• 《网络编程 —— 实时判断socket连接状态?》


    • 对端正常close socket,或者进程退出(正常退出或崩溃),对端系统正常关闭
    这种情况下,协议栈会走正常的关闭状态转移,使用epoll的话,一般要判断如下几个情况
    • 处理可读事件时,在循环read后,返回结果为0
    • 处理可写事件时,write返回-1,errno为EPIPE
    • EPOLLERR或EPOLLHUP或事件
    • 对端非正常断开,比如服务器断电,网线被拔掉
    这种情况下,协议栈无法感知,SO_KEEPALIVE这个选项的超时事件太长并不实用,一般还是以应用层的heartbeat来及时发现。(若使用,参考法四)

    下面来罗列一下判断远端已经断开的方法:

    法一:

    当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 断开

    函数封装,单独使用(没有使用read、write、select时),判读网络是否断开。 

    法三:

    若使用了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错误

    任何一端有数据发送就不会启动keepalive(keepalive无效,不起作用),参考:通过设置TCP_USER_TIMEOUT参数来判断tcp连接是否断开

    复制代码
        unsigned int timeout = 1000;   //值为数据包被发送后未接收到ACK确认的最大时长,以毫秒
                                        //为单位,例如设置为timeout时,代表如果发送出去的数据包
                                        //在timeout毫秒内未收到ACK确认,则下一次调用send或者recv,
                                        //函数会返回-1,errno设置为ETIMEOUT,代表connection timeout。
    
        //使用TCP Keep-alive加上TCP_USER_TIMEOUT机制,就可以完美解
        //决通信对端异常断网、掉电的情况下,连接被长期挂起的问题了。
        setsockopt(socket_fd, IPPROTO_TCP, 18, &timeout, sizeof(timeout));
        //setsockopt(socket_fd, IPPROTO_TCP, TCP_USER_TIMEOUT, &timeout, sizeof(timeout));
    复制代码

    法五:

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

  • 相关阅读:
    构建安全的Xml Web Service系列之如何察看SoapMessage
    web2.0盈利模式
    Asp.Net ajax v1.0莫名出现"Sys未定义"的原因
    北京街头发生一幕~让人深思!!!
    算法函数:得到一个字符串中的最大长度的数字
    手把手教你如何扩展GridView之自动排序篇
    手把手教你如何扩展GridView之自带Excel和Word导出
    nhibernate学习之集合组合依赖
    递归算法学习系列一(分而治之策略)
    手把手教你如何扩展GridView之自带分页
  • 原文地址:https://www.cnblogs.com/zhuangquan/p/16282143.html
Copyright © 2020-2023  润新知