日期:2014-05-16  浏览次数:21144 次

Linux下keepalive为什么不起作用
测试时发现,网线拔掉后,如果该socket一直在尝试接收,且不发送数据,则10秒左右能够收到断开信号。
但是如果socket一直在发送数据,则keepalive不起作用,直到发送缓冲区满了才能断开

int keepalive = 1; // 开启keepalive属性
int keepidle = 5; // 如该连接在5秒内没有任何数据往来,则进行探测
int keepinterval = 1; // 探测时发包的时间间隔为1 秒
int keepcount = 5; // 探测尝试的次数.如果第1次探测包就收到响应了,则后4次的不再发
socklen_t optlen = sizeof(socklen_t);
if (setsockopt(m_Socket, SOL_SOCKET, SO_KEEPALIVE, (void *)&keepalive, optlen) < 0 ||
setsockopt(m_Socket, SOL_TCP, TCP_KEEPIDLE, (void *)&keepidle, optlen) < 0 ||
setsockopt(m_Socket, SOL_TCP, TCP_KEEPINTVL, (void *)&keepinterval, optlen) < 0 ||
setsockopt(m_Socket, SOL_TCP, TCP_KEEPCNT, (void *)&keepcount, optlen) < 0)
{
ClosePort();
return false;
}

------解决方案--------------------
按你的现象分析感觉好像起作用了吧:如果没有设置keep alive,可能你在你的socket(阻塞性)的上面,接收: recv会一直阻塞不能返回,除非对端主动关闭连接,因为recv不知道socket断了。
------解决方案--------------------
使能此选项,允许TCP套接口在TCP连接上发送“保活”数据包。此选项的设置值保存在传输控制块的sk_flags的SOCK_KEEPOPEN位,同时根据是否启用了保活功能来确定是否开启保活定时器。
保活定时器在应用进程选取了套接口SO_KEEPALIVE选项时生效。如果连接的连续空闲时间超过2小时,则保活定时器超时,向对端发送连接探测报文,强迫对端响应。