背景
项目中经常使用LVS做负载,当一个长连接过段时间不发消息时,LVS之后就不会再转发这个长连接的TCP数据包。
使用TCP长连接的场景
一种情况:为了节省TCP连接时间,我们选择建立几个固定的TCP长连接,然后把所有的消息平均分配的这几个TCP通道上传送给对方。
另一种情况:客户端和服务端之间是一个会话,要隔一段时间发个心跳。
现在出现了第三种情况,我是一个代理程序,要转发一个会话,让服务端和客户端都无感知;而且多个会话可以复用一个通路
(1)我需要在会话建立之前就建立连接;
(2)没设计代理的心跳接口,我不能自己给后端发心跳;
但代理和后端之间隔着LVS。我不发心跳,LVS就偷偷地把长连接断开了,而此时代理和后端都一无所知。从连接上看都是ESTABLISHED,一发消息就被RST。
这时我们就用到了TCP自带的keepalive机制。
配置TCP长连接参数
在Centos7上的尝试,配置以下内核参数
# cat /etc/sysctl.conf
net.ipv4.tcp_keepalive_time = 30
net.ipv4.tcp_keepalive_probes = 2
net.ipv4.tcp_keepalive_intvl = 20
但是没见生效;为什么没生效?因为需要自己在程序里开启KEEPALIVE标识;(也就是说得让系统知道你是长连接)
在C++的程序中开启方法如下:
int keepAlive = 5;
int keepIdle = 5;
int keepInterval = 5;
int keepCount = 3;
if(setsockopt(fd,SOL_SOCKET,SO_KEEPALIVE,(void*)&keepAlive,sizeof(keepAlive)) == -1)
{
printf("setsockopt SO_KEEPALIVE error!");
}
思考问题:那发送间隔等参数的设置,最后以谁为准呢?
答:如果程序中设置了,以程序中的为准。如果只是像上面开启了标识,没设置参数。那么就以/etc/sysctl.conf中配置的参数为准。
转载请注明来源:https://www.cnblogs.com/bugutian/