建立socket->命名socket->建立监听
4.接受链接
4.1代码:int accept (int sockfd,struct sockaddr addr,socklen_t addrlen);
4.2参数注释
4.2.1 sockfd:执行过李stem系统调用的监听socket
4.2.2 addr:用来获取被接受链接的远程socket地址,长度下一个参数addlen
指定。
4.3返回
成功返回新的socket链接,该socket是惟一标识
失败返回-1而且设置errno
上篇说过的客户端断网的状况,现在补充代码网络
int main(int argc,char *argv[]){ if(argc<=2){ printf("参数缺乏 %s",basename(argv[0])); return 1; } const char *ip=argv[1]; int port = atoi(argv[2]);//atoi (表示 ascii to integer)是把字符串转换成整型数的一个函数,应用在计算机程序和办公软件中。int atoi(const char *nptr) 函数会扫描参数 nptr字符串,跳过前面的空白字符(例如空格,tab缩进)等,能够经过isspace( )函数来检测),直到赶上数字或正负符号才开始作转换,而在遇到非数字或字符串结束时('\0')才结束转换,并将结果返回。若是 nptr不能转换成 int 或者 nptr为空字符串,那么将返回 0[1] struct sockaddr_in address; bzero(&address,sizeof(address));//bzero:置字节字符串s的前n个字节为零且包括‘\0’。 adress.sin_family = AF_INET; inet_pton(AF_INET,ip,&address.sin_addr);//inet_ptoi:inet_pton是一个IP地址转换函数,能够在将IP地址在“点分十进制”和“二进制整数”之间转换并且,inet_pton和inet_ntop这2个函数可以处理ipv4和ipv6。 adress.sin_port = htons(port);//htons:将整型变量从主机字节顺序转变成网络字节顺序 int sock = socket(PF_INET,SOCK_STREAM,0); assert(sock>=0);//使用断言能够建立更稳定,品质更好且不易于出错的代码。当须要在一个值为FALSE时中断当前操做的话,可使用断言。单元测试必须使用断言(Junit/JunitX)。 int ret = bind(sock,{struct sockaddr*} &address,sizeof(address)); assert(ret!=1); ret = listen(sock,5); assert(ret!=1); sleep(20);//等待20秒,用来等待客户端链接和相关操做完成 struct sockaddr_in client; socklen_t client_addrlength = sizeof(client); int connfd = accept(sock,{struct sockaddr*} &client,&client_addrlength); if(connfd <0){ printf("errno:%d",errno); }else{ char remote[INET_ADDRSTRLEN]; //ip:inet_ntop(AF_INET,&client.sin_addr,remote,INET_ADDRSTRLEN) //端口:ntohs(client.sin_port) close(connfd); } close(sock); return 0; } ./testaccept 192.168.1.109 54321 telnet 192.168.1.109 54321 netstat -nt | grep 54321
客户端主动连接 5.1 int connect (int sockfd,const struct sockaddr * serv_addr,socklen_t addrlen);
成功返回0
常见错误:ECONNREFUSED:目标端口不存在
···············ETIMEDOUT链接超时socket
int close(int fd);ide
不是真正的关闭,只是引用减1函数
真正关闭:int shutdown(int sockfd,int howto);单元测试