【转】 linux应用程序: tcp检测断开机制

传送门:https://blog.csdn.net/qqliyunpeng/article/details/79821482

              https://www.cnblogs.com/tdyizhen1314/p/4324441.html

法一:

当recv()返回值小于等于0时,socket连接断开。但是还需要判断 errno是否等于 EINTR,如果errno == EINTR 则说明recv函数是由于程序接收到信号后返回的,socket连接还是正常的,不应close掉socket连接。

 

法二:

  struct tcp_info info; 
  int len=sizeof(info); 
  getsockopt(sock, IPPROTO_TCP, TCP_INFO, &info, (socklen_t *)&len); 
  if((info.tcpi_state==TCP_ESTABLISHED))  则说明未断开  else 断开

 

法三:

若使用了select等系统函数,若远端断开,则select返回1,recv返回0则断开。其他注意事项同法一。

 

法四:

int keepAlive = 1; // 开启keepalive属性
int keepIdle = 60; // 如该连接在60秒内没有任何数据往来,则进行探测 
int keepInterval = 5; // 探测时发包的时间间隔为5 秒
int keepCount = 3; // 探测尝试的次数.如果第1次探测包就收到响应了,则后2次的不再发.

setsockopt(rs, SOL_SOCKET, SO_KEEPALIVE, (void *)&keepAlive, sizeof(keepAlive));
setsockopt(rs, SOL_TCP, TCP_KEEPIDLE, (void*)&keepIdle, sizeof(keepIdle));
setsockopt(rs, SOL_TCP, TCP_KEEPINTVL, (void *)&keepInterval, sizeof(keepInterval));
setsockopt(rs, SOL_TCP, TCP_KEEPCNT, (void *)&keepCount, sizeof(keepCount));

设置后,若断开,则在使用该socket读写时立即失败,并返回ETIMEDOUT错误

 

法五:

自己实现一个心跳检测,一定时间内未收到自定义的心跳包则标记为已断开。

代码实现:

1.判断sock的属性

int tcp_is_connected(int fd)
{
    struct tcp_info info;
    int len = sizeof(info);

    if (fd <= 0) return 0;

    getsockopt(fd, IPPROTO_TCP, TCP_INFO, &info, (socklen_t *)&len);
    /*if (info.tcpi_state == TCP_ESTABLISHED) */
    if (info.tcpi_state == 1)  // Is connected
        return 1;
    else  // Losed
        return 0;
}
--------------------- 
作者:qqliyunpeng 
来源:CSDN 
原文:https://blog.csdn.net/qqliyunpeng/article/details/79821482 
版权声明:本文为博主原创文章,转载请附上博文链接!

2.keepalive

int SetKeepAlive(int fd, int idle, int cnt, int intv)
{  
    int alive;
    /*int flag, idle, cnt, intv;  */

    /* Set: use keepalive on fd */  
    alive = 1;  
    if (setsockopt(fd, SOL_SOCKET, SO_KEEPALIVE, &alive,  sizeof alive) != 0)  
    {  
        perror("Set keepalive fail");  
        return -1;  
    }  

    /* idle秒钟无数据,触发保活机制,发送保活包 */  
    /*idle = 10;  */
    if (setsockopt (fd, SOL_TCP, TCP_KEEPIDLE, &idle, sizeof idle) != 0)  
    {  
        perror("Set keepalive idle fail");  
        return -1;  
    }  

    /* 如果没有收到回应,则intv秒钟后重发保活包 */  
    /*intv = 5;  */
    if (setsockopt (fd, SOL_TCP, TCP_KEEPINTVL, &intv, sizeof intv) != 0)  
    {  
        perror("Set keepalive intv fail");  
        return -1;  
    }  

    /* 连续cnt次没收到保活包,视为连接失效 */  
    /*cnt = 3;  */
    if (setsockopt (fd, SOL_TCP, TCP_KEEPCNT, &cnt, sizeof cnt) != 0)  
    {  
        perror("Set keepalive cnt fail");  
        return -1;  
    }  

    return 0;  
}
--------------------- 
作者:qqliyunpeng 
来源:CSDN 
原文:https://blog.csdn.net/qqliyunpeng/article/details/79821482 
版权声明:本文为博主原创文章,转载请附上博文链接!

采用SetKeepAlive(gtcp_set.fd, 6, 2, 2)函数,并使用wireshark抓取的断开的一段时间的数据如下:

从左到右列表示的意思依次是时间,原地址,目的地址,协议。。

我们看到,图上的①②③经历的时间都是6s,这是正常的keepalive询问,网线还没有断开的状态,③之后是网线断开,当网线断开192.168.15.11在194s发送询问,192.168.15.110并没有回复数据, 2s之后196s发送询问,依然没有回复,第二次2s后198,依然没有回复,视为连路断开。
--------------------- 
作者:qqliyunpeng 
来源:CSDN 
原文:https://blog.csdn.net/qqliyunpeng/article/details/79821482 
版权声明:本文为博主原创文章,转载请附上博文链接!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值