背景

项目中经常使用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中配置的参数为准。

posted on 2020-05-22 20:09  步孤天  阅读(10307)  评论(0编辑  收藏  举报