Tcp-KeepAlive保活机制
在TCP连接中存在心跳包的机制,自动的在规定的时间内向对方发送心跳包来检测对方是否仍然在线。对方收到心跳包会进行回复以告知自己在线。也就是说可以开启SO_KEEPALIVE选项。Linux下Kernel有三个参数影响到KeepAlive的行为。
- tcp_keepalive_time 7200// 当间隔7200s未收到对方数据则开始进行检测
- tcp_keepalive_intvl 75 //检测开始每多少时间发送心跳包,单位秒,默认75s,即每隔75s发送一次心跳包
- tcp_keepalive_probe //发送几次心跳包对方未响应则关闭连接,默认9次
开启KeepAlive功能需要消耗额外的宽带和流量,所以TCP协议层默认并不开启KeepAlive功能,需要自己手动设置。另外默认的KeepAlive超时需要7,200Seconds, 即2小时,探测次数为9次。对于很多服务端应用程序来说,2小时的空闲时间太长。因此,我们需要手工开启KeepAlive功能并设置合理的KeepAlive参数。
TCP socket也有三个选项和内核对应,通过setsockopt系统调用针对单独的socket进行设置:
- TCP_KEEPCNT:覆盖 tcp_keepalive_probes
- TCP_KEEPIDLE: 覆盖 tcp_keep_alive_time
- TCP_KEEPINTVL:覆盖 tcp_keep_alive_intvl
具体设置方法(关键代码):
int sockfd;
struct sockaddr_in sin;
int optval;
socklen_t optlen = sizeof(optval);
sockfd=socket(AF_INET,SOCK_STREAM,0);
sin.sin_family=AF_INET;
sin.sin_addr.s_addr=addr.sin_addr.s_addr;
sin.sin_port=htons(PORT);
optval = 1;
setsockopt(sockfd, SOL_SOCKET, SO_KEEPALIVE, &optval, optlen);
optval = 5;
setsockopt(sockfd, SOL_TCP, TCP_KEEPCNT, &optval, optlen);
optval = 1;
setsockopt(sockfd, SOL_TCP, TCP_KEEPIDLE, &optval, optlen);
optval = 1;
setsockopt(sockfd, SOL_TCP, TCP_KEEPINTVL, &optval, optlen);
connect(sockfd,(struct sockaddr *)&sin,sizeof(sin));```
参考链接
KeepAlive详解 网络之心跳机制
|