三次握手
TCP连接建立的开始是三次握手,通过三次交互确认连接成功,在客户端调用connect时,客户端发送sync消息给服务端,服务端收到sync消息后,返回一个ack+sync,并等待ack,客户端收到ack+sync后,返回一个ack,connect返回,服务端收到ack后,accept返回,如下图所示:
connect超时设置
如果connect连接的服务端不存在,或是异常了,会出现什么情况,可能有以下几种情况:
-
服务端返回连接错误
-
等待系统默认的75秒超时
上述两种情况都是不可控的,因此可以通过设置超时时间的方式控制超时时间,有下述两种方法。
非阻塞connect+select
将conenct设置成非阻塞后,connect在发送sync后直接返回-1,然后使用select等待ack+sync,select是可以设置超时时间,因此达到设置超时时间的目的。注意需要将socket恢复成阻塞模式,如下所示:
dwSign = 1; if((err = ioctl(socket, FIONBIO, &dwSign)) < 0) { printf("ioctl socket failed, errno = %d! ", err); close(socket); return err; } SOCKADDR_IN tServerAddrInfo = {0}; tServerAddrInfo.sin_family = AF_INET; tServerAddrInfo.sin_addr.s_addr = ServerIP; tServerAddrInfo.sin_port = ServerPort; int bRet = 0x0; fd_set set = {0x0}; if(-1 == connect(socket, (SOCKADDR*)&tServerAddrInfo, sizeof(SOCKADDR))) { tm.tv_sec = 10; tm.tv_usec = 0; FD_ZERO(&set); FD_SET(socket, &set); // select返回值大于0,表示有数据可以操作;select返回值等于0,表示超时 if(select(socket + 1, NULL, &set, NULL, &tm) > 0) { // 获取socket的error值,0表示成功 getsockopt(socket, SOL_SOCKET, SO_ERROR, &error, (socklen_t *)&len); if(error == 0) { bRet = TRUE; } else { bRet = FALSE; } } else { bRet = FALSE; } } else { bRet = TRUE; } dwSign = 0; if(0 > (err = ioctl(socket, FIONBIO, &dwSign))) { printf("ioctl socket failed, errno = %d! ", err); close(dwSocket); return err; }
设置socket的发送超时时间
调用connect函数实际上是发送sync报文,如果设置发送超时时间,那么也就可以实现connect的超时时间,如下所示:
struct timeval timeout = {10, 0); setsockopt(socket, SOL_SOCKET, SO_SNDTIMEO, (char *)&timeout, sizeof(struct timeval));
accpet超时设置
既然可以通过设置发送超时时间,控制connect的超时时间,那么同理,也可以通过设置接收超时时间来设置accept的超时时间,如下所示:
struct timeval timeout = {10, 0}; setsockopt(proxy_sock, SOL_SOCKET, SO_RCVTIMEO, (char *)&timeout, sizeof(struct timeval));