代码改变世界
[登录 · 注册]
  • 如何在C语言中判断socket是否已经断开
  • 下面来介绍判断非阻塞SOCKET是否已经断开的几种方法:

    注意要区分不同操作系统分别进行测试, 包括WINDOWS, LINUX和UNIX会各有不同。

    在WINDOWS下比较简单,可以使用FD_CLOSE事件判断SOCKET是否已经断开

    view plaincopy to clipboardprint?
    bool IsSocketClosed(SOCKET clientSocket)
    {
     bool ret = false;
     HANDLE closeEvent = WSACreateEvent();
     WSAEventSelect(clientSocket, closeEvent, FD_CLOSE); 
    
     DWORD dwRet = WaitForSingleObject(closeEvent, 0); 
    
     if(dwRet == WSA_WAIT_EVENT_0)
     ret = true;
     else if(dwRet == WSA_WAIT_TIMEOUT)
     ret = false; 
    
     WSACloseEvent(closeEvent);
     return ret;
    }
     

    在UNIX/LINUX下,非阻塞模式SOCKET可以采用recv+MSG_PEEK的方式进行判断,其中MSG_PEEK保证了仅仅进行状态判断,而不影响数据接收

    对于主动关闭的SOCKET, recv返回-1,而且errno被置为9(#define EBADF   9 /* Bad file number */)

    或104 (#define ECONNRESET 104 /* Connection reset by peer */)

    对于被动关闭的SOCKET,recv返回0,而且errno被置为11(#define EWOULDBLOCK EAGAIN /* Operation would block */)

    对正常的SOCKET, 如果有接收数据,则返回>0, 否则返回-1,而且errno被置为11(#define EWOULDBLOCK EAGAIN /* Operation would block */)

    因此对于简单的状态判断(不过多考虑异常情况),
        recv返回>0,   正常
        返回-1,而且errno被置为11  正常
        其它情况    关闭

    view plaincopy to clipboardprint?
    #include <errno.h> 
    
    bool IsSocketClosed(int clientSocket)
    {
     char buff[32];
     int recvBytes = recv(clientSocket, buff, sizeof(buff), MSG_PEEK); 
    
     int sockErr = errno; 
    
     //cout << "In close function, recv " << recvBytes << " bytes, err " << sockErr << endl; 
    
     if( recvBytes > 0) //Get data
     return false; 
    
     if( (recvBytes == -1) && (sockErr == EWOULDBLOCK) ) //No receive data
     return false; 
    
     return true;
    }
    

    此外,所有操作系统上还可以通过TCP_KEEPLIVE实行心跳检测

  • 下一篇:获取视频文件格式信息的工具和方法
  • 【推广】 阿里云小站-上云优惠聚集地(新老客户同享)更有每天限时秒杀!
    【推广】 云服务器低至0.95折 1核2G ECS云服务器8.1元/月
    【推广】 阿里云老用户升级四重礼遇享6.5折限时折扣!
  • 原文:https://www.cnblogs.com/lidabo/p/12981571.html
走看看 - 开发者的网上家园