socket选项自带的TCP异常断开检测

  TCP异常断开是指在突然断电,直接拔网线等等情况下,如果通信双方没有进行数据发送通信等处理的时候,无法获知连接已经断开的情况.

  在通常的情况下,为了使得socket通信不受操作系统的限制,需要自己在应用层实现心跳包机制,来检查异常断开的情况,一般的方式就是服务器在一段时间没有收到客户端数据包时,定时发包,然后客户端回应,如果已经出现异常断开则服务器接收会返回错误,而客户端在指定时间内没有收到数据包,则主动向服务器发包,得到错误就说明断开.诸如此类的方式就是自己实现的心跳包机制.

  但操作系统本身也自带了一些心跳包机制,这些机制是由socket的TCP栈底层实现的,不会影响应用层通信,也不需要应用层自己处理,发现异常断开可以自行检查出来并返回错误(它的本质也是在空闲时发送心跳包).以下介绍一下Windows以及linux下的方法.

  首先介绍Windows下的方法,该方式要求通信双方必须都是Windows NT以上操作系统(如果是其它版本操作系统,如linux等等,不敢保证100%无效).MSDN中有描述WSAIoctl中的SIO_KEEPALIVE_VALS选项,该选项以及struct tcp_keepalive的定义在MSTCPiP.h有,不进行说明了,直接看代码:

 

    #include <MSTCPiP.h>

 

    DWORD retBytes;
    tcp_keepalive inKeepSetting;
    tcp_keepalive retKeepSetting;

    inKeepSetting.onoff = 1;                //探测次数
    inKeepSetting.keepalivetime = 5500;        // 首次探测开始前的tcp无数据收发空闲时间
    inKeepSetting.keepaliveinterval = 3000;    // 每次探测的间隔时间

    if (WSAIoctl(aptSock, SIO_KEEPALIVE_VALS,
        &inKeepSetting,
        sizeof(inKeepSetting),
        &retKeepSetting,
        sizeof(retKeepSetting),
        &retBytes,
        NULL,
        NULL) != 0)
    {
        printf("WSAIoctl Error: %d/n", WSAGetLastError());
    }

 

  Linux下的方式是通过setsockopt来设置选项,见代码(代码从网络上摘录了部分):

  #include   <netinet/tcp.h>
  ……

 

  #define SOCKET_ERROR (-1)

 

  // 以秒为单位
  int   keepAlive = 1;       //设定KeepAlive
  int   keepIdle = 5;        //首次探测开始前的tcp无数据收发空闲时间
  int   keepInterval = 3;  //每次探测的间隔时间
  int   keepCount = 2;     //探测次数
               
  if(setsockopt(s,SOL_SOCKET,SO_KEEPALIVE,(void*)&keepAlive,sizeof(keepAlive)) == SOCKET_ERROR)

      printf("Call setsockopt error, errno is %d/n", errno);


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

      printf("Call setsockopt error, errno is %d/n", errno);

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

      printf("Call setsockopt error, errno is %d/n", errno);
               
  if(setsockopt(s,SOL_TCP,TCP_KEEPCNT,(void   *)&keepCount,sizeof(keepCount)) == SOCKET_ERROR)

      printf("Call setsockopt error, errno is %d/n", errno);

  • 1
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Linux下,可以使用TCP keepalive机制来检测TCP连接是否断开TCP keepalive是TCP协议的一个选项,可以让操作系统在连接空闲一段时间后发送一些数据包来检测连接是否仍然有效。 使用TCP keepalive机制需要先设置keepalive选项,可以使用setsockopt函数来设置。以下是示例代码: ```c int keepalive = 1; // 开启keepalive选项 int keepidle = 60; // 连接空闲60秒后开始发送keepalive包 int keepinterval = 5; // 每5秒发送一次keepalive包 int keepcount = 3; // 发送3次keepalive包后仍未收到回应则认为连接断开 setsockopt(sockfd, SOL_SOCKET, SO_KEEPALIVE, (void *)&keepalive, sizeof(keepalive)); setsockopt(sockfd, SOL_TCP, TCP_KEEPIDLE, (void *)&keepidle, sizeof(keepidle)); setsockopt(sockfd, SOL_TCP, TCP_KEEPINTVL, (void *)&keepinterval, sizeof(keepinterval)); setsockopt(sockfd, SOL_TCP, TCP_KEEPCNT, (void *)&keepcount, sizeof(keepcount)); ``` 设置完keepalive选项后,操作系统会在连接空闲一段时间后开始发送keepalive包。如果远端主机在一定时间内没有回复这些包,操作系统会认为连接已经断开,然后关闭socket。可以通过捕捉SIGPIPE信号来处理连接断开的事件。 ```c struct sigaction sa; sa.sa_handler = SIG_IGN; sigaction(SIGPIPE, &sa, 0); ``` 在实际应用中,可以使用select或者epoll等函数来监听socket的可读事件和可写事件。如果可读事件返回0或者可写事件返回错误,则可以认为连接已经断开

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值