linux tcp 心跳程序,tcp连接探测Keepalive和心跳包

MaxDataRetries"="5"

对于实用的程序来说,2小时的空闲时间太长。因此,我们需要手工开启Keepalive功能并设置合理的Keepalive参数。

cbef093dcc044b2793832001e2365e43.png//开启KeepAlivecbef093dcc044b2793832001e2365e43.pngBOOL bKeepAlive=TRUE;

cbef093dcc044b2793832001e2365e43.pngintnRet=::setsockopt(socket_handle, SOL_SOCKET, SO_KEEPALIVE, (char*)&bKeepAlive,sizeof(bKeepAlive));

cbef093dcc044b2793832001e2365e43.png

cbef093dcc044b2793832001e2365e43.pngif(nRet==SOCKET_ERROR)

2f88ce130b654eb5dc6788e02dbcfc90.png

dbf989d57862681739b642d8621fe1f0.png918e8df969f9f8c8d002f25cda86cade.png{

df37983f39daa189b8c814e01a6a9011.pngreturnFALSE;

0ac3a2d53663ec01c7f7225264eeefae.png}cbef093dcc044b2793832001e2365e43.png

cbef093dcc044b2793832001e2365e43.png//设置KeepAlive参数2f88ce130b654eb5dc6788e02dbcfc90.png

dbf989d57862681739b642d8621fe1f0.pngtcp_keepalive alive_in=918e8df969f9f8c8d002f25cda86cade.png{0};

2f88ce130b654eb5dc6788e02dbcfc90.png

dbf989d57862681739b642d8621fe1f0.pngtcp_keepalive alive_out=918e8df969f9f8c8d002f25cda86cade.png{0};

cbef093dcc044b2793832001e2365e43.pngalive_in.keepalivetime=5000;//开始首次KeepAlive探测前的TCP空闭时间cbef093dcc044b2793832001e2365e43.pngcbef093dcc044b2793832001e2365e43.pngalive_in.keepaliveinterval=1000;//两次KeepAlive探测间的时间间隔cbef093dcc044b2793832001e2365e43.pngcbef093dcc044b2793832001e2365e43.pngalive_in.onoff=TRUE;

cbef093dcc044b2793832001e2365e43.pngunsignedlongulBytesReturn=0;

cbef093dcc044b2793832001e2365e43.png

cbef093dcc044b2793832001e2365e43.pngnRet=WSAIoctl(socket_handle, SIO_KEEPALIVE_VALS,&alive_in,sizeof(alive_in),

cbef093dcc044b2793832001e2365e43.png&alive_out,sizeof(alive_out),&ulBytesReturn, NULL, NULL);

cbef093dcc044b2793832001e2365e43.pngif(nRet==SOCKET_ERROR)

2f88ce130b654eb5dc6788e02dbcfc90.png

dbf989d57862681739b642d8621fe1f0.png918e8df969f9f8c8d002f25cda86cade.png{

df37983f39daa189b8c814e01a6a9011.pngreturnFALSE;

df37983f39daa189b8c814e01a6a9011.png

df37983f39daa189b8c814e01a6a9011.png

0ac3a2d53663ec01c7f7225264eeefae.png}cbef093dcc044b2793832001e2365e43.png

cbef093dcc044b2793832001e2365e43.png

开启Keepalive选项之后,对于使用IOCP模型的服务器端程序来说,一旦检测到连接断开,GetQueuedCompletionStatus函数将立即返回FALSE,使得服务器端能及时清除该连接、释放该连接相关的资源。对于使用select模型的客户端来说,连接断开被探测到时,以recv目的阻塞在socket上的select方法将立即返回SOCKET_ERROR,从而得知连接已失效,客户端程序便有机会及时执行清除工作、提醒用户或重新连接。

另一种技术,由应用程序自己发送心跳包来检测连接的健康性。客户端可以在一个Timer中或低级别的线程中定时向发服务器发送一个短小精悍的包,并等待服务器的回应。客户端程序在一定时间内没有收到服务器回应即认为连接不可用,同样,服务器在一定时间内没有收到客户端的心跳包则认为客户端已经掉线。

++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

windows下此处的”非正常断开”指TCP连接不是以优雅的方式断开,如网线故障等物理链路的原因,还有突然主机断电等原因.

有两种方法可以检测:

1.TCP连接双方定时发握手消息

2.利用TCP协议栈中的KeepAlive探测

第二种方法简单可靠,只需对TCP连接两个Socket设定KeepAlive探测,

所以本文只讲第二种方法在Linux,Window2000下的实现(在其它的平台上没有作进一步的测试)

cbef093dcc044b2793832001e2365e43.pngcbef093dcc044b2793832001e2365e43.pngWindows 2000平台下 头文件

cbef093dcc044b2793832001e2365e43.png#includecbef093dcc044b2793832001e2365e43.png//定义结构及宏

/*2f88ce130b654eb5dc6788e02dbcfc90.png

dbf989d57862681739b642d8621fe1f0.pngstructTCP_KEEPALIVE918e8df969f9f8c8d002f25cda86cade.png{

df37983f39daa189b8c814e01a6a9011.pngu_longonoff;

df37983f39daa189b8c814e01a6a9011.pngu_longkeepalivetime;

df37983f39daa189b8c814e01a6a9011.pngu_longkeepaliveinterval;

0ac3a2d53663ec01c7f7225264eeefae.png};

cbef093dcc044b2793832001e2365e43.png*/

cbef093dcc044b2793832001e2365e43.pngtcp_keepalive live,liveout; 

live.keepaliveinterval=5000; //每5秒发一次探测报文,发5次没有回应,就断开

live.keepalivetime=30000;//超过30s没有数据,就发送控测包

live.onoff=TRUE;

int Opt = 1;

int iRet = setsockopt(Accept,SOL_SOCKET,SO_KEEPALIVE,(char *)&Opt,sizeof(int));

if(iRet == 0){

DWORD dw;

if(::WSAIoctl(Accept,SIO_KEEPALIVE_VALS,

&live,sizeof(live),&liveout,sizeof(liveout),

&dw,NULL,NULL)== SOCKET_ERROR){

}

}

cbef093dcc044b2793832001e2365e43.png

cbef093dcc044b2793832001e2365e43.png

ACE下代码 //by rainfish blog.csdn.net/bat603

int Opt = 1;

//在测试过程中,发现检测的次数是5次,即下面的设置中,从最近一次消息开始计算的10秒后,每次间隔5秒,连续发送5次,即35秒发现网络断了

tcp_keepalive live,liveout;

live.keepaliveinterval=5000; //每次检测的间隔 (单位毫秒)

live.keepalivetime=10000; //第一次开始发送的时间(单位毫秒)

live.onoff=TRUE;

int iRet = stream.set_option(SOL_SOCKET,SO_KEEPALIVE,&Opt,sizeof(int));

if(iRet == 0){

DWORD dw;

//此处显示了在ACE下获取套接字的方法,即句柄的(SOCKET)化就是句柄

if(WSAIoctl((SOCKET)h,SIO_KEEPALIVE_VALS,&live,sizeof(live),

&liveout,sizeof(liveout),&dw,NULL,NULL)== SOCKET_ERROR){

//Delete Client

return;

}

}

Linux平台下

#include "/usr/include/linux/tcp.h"

#include "/usr/include/linux/socket.h"

KeepAlive实现,单位秒

//下面代码要求有ACE,如果没有包含ACE,则请把用到的ACE函数改成linux相应的接口

int keepAlive = 1;//设定KeepAlive

int keepIdle = 5;//开始首次KeepAlive探测前的TCP空闭时间

int keepInterval = 5;//两次KeepAlive探测间的时间间隔

int keepCount = 3;//判定断开前的KeepAlive探测次数

if(setsockopt(s,SOL_SOCKET,SO_KEEPALIVE,(void*)&keepAlive,sizeof(keepAlive)) == -1)

{

ACE_DEBUG ((LM_INFO,

ACE_TEXT ("(%P|%t) setsockopt SO_KEEPALIVE error!/n")));

}

if(setsockopt(s,SOL_TCP,TCP_KEEPIDLE,(void *)&keepIdle,sizeof(keepIdle)) == -1)

{

ACE_DEBUG ((LM_INFO,

ACE_TEXT ("(%P|%t) setsockopt TCP_KEEPIDLE error!/n")));

}

if(setsockopt(s,SOL_TCP,TCP_KEEPINTVL,(void *)&keepInterval,sizeof(keepInterval)) == -1)

{

ACE_DEBUG ((LM_INFO,

ACE_TEXT ("(%P|%t) setsockopt TCP_KEEPINTVL error!/n")));

}

if(setsockopt(s,SOL_TCP,TCP_KEEPCNT,(void *)&keepCount,sizeof(keepCount)) == -1)

{

ACE_DEBUG ((LM_INFO,

ACE_TEXT ("(%P|%t)setsockopt TCP_KEEPCNT error!/n")));

}

++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值