最近在写一个TCP服务器端的程序的时候,出现一个问题,客户的内网穿的NAT比较,网络比较复杂,出现了TCP客户端已经死掉或者离线的情况下,但服务器的TCP连接状态还是CONNECT的,为了这个问题的解决方法也是在网上查了很多,有的人说用心跳保活,如果服务器端多少时间没有收到客户端发过来的消息,就认为客户端已经掉线,close掉这个连接socket,但是这个方法的工作量比较大,最后采用了TCP/IP自带的setsockopt函数
使用setsockopt要包含sys/socket.h这个头文件
int getsockopt(int sockfd, int level, int optname,
void *optval, socklen_t *optlen);
int setsockopt(int sockfd, int level, int optname,
const void *optval, socklen_t optlen);
使用setsockopt和getsockopt这两个函数要包含sys/socket.h这个头文件。
1.int keepAlive = 1; // 开启keepalive属性. 缺省值: 0(关闭)
2.int keepIdle = 60; // 如果在60秒内没有任何数据交互,则进行测试. 缺省值:7200(s)
3.int keepInterval = 5; // 探测时发探测包的时间间隔为5秒. 缺省值:75(s)
4.int keepCount = 2; // 探测重试的次数. 全部超时则认定连接失效..缺省值:9(次)
5.setsockopt(s, SOL_SOCKET, SO_KEEPALIVE, (void*)&keepAlive, sizeof(keepAlive));
6.setsockopt(s, SOL_TCP, TCP_KEEPIDLE, (void*)&keepIdle, sizeof(keepIdle));
7.setsockopt(s, SOL_TCP, TCP_KEEPINTVL, (void*)&keepInterval, sizeof(keepInterval));
8.setsockopt(s, SOL_TCP, TCP_KEEPCNT, (void*)&keepCount, sizeof(keepCount));
你也可以只使用setsockopt(s, SOL_SOCKET, SO_KEEPALIVE, (void *)&keepalive, sizeof(keepalive));
这个只表示开始了TCP保活,后面的三个函数不调用的话会自动使用系统的默认值,系统的默认值文件路径为
/proc/sys/net/ipv4/tcp_keepalive_time
/proc/sys/net/ipv4/tcp_keepalive_intvl
/proc/sys/net/ipv4/tcp_keepalive_probes