关于socket判断连接断开并重连

来源:互联网 发布:飞升功法升级数据 编辑:程序博客网 时间:2024/05/16 14:57

关于socket判断连接断开并重连 以下是个线程函数 UINT CMonitorDlg::ThreadFunction(LPVOID pParam) { WSADATA wsaData; SOCKET s; //创建套接字 int port=PORT; int iLen; //从服务器接收的数据长度 TCHAR recvbuf[BufferLen]; //接受数据的缓冲器 struct sockaddr_in serv1; //服务器端地址 serv1.sin_family=AF_INET; //需要连接的服务器地址信息 serv1.sin_port=htons(port); //需要连接的服务器地址信息 serv1.sin_addr.s_addr=inet_addr(p->Ip); //将命令行的IP地址转化为二进制表示的网络字节顺序IP地址 s=socket(AF_INET,SOCK_STREAM,0); if(s==INVALID_SOCKET) { AfxMessageBox("socket()failed");//套接字失败 return 0; } if(connect(s,(struct sockaddr*)&serv1,sizeof(serv1))==INVALID_SOCKET) { int b=WSAGetLastError(); //连接失败 } else { //连接成功 p->connectstate=true; do{ 在此循环体中我接收数据并对数据做处理(省略部分代码) iLen=recv(s,recvbuf,sizeof(recvbuf),0); //接收套接字中的数据放入recvbuf1缓冲区 if(iLen==0) { strtempMsg.Format(_T("局 %s 数据传送结束"),p->Name); return 0; } else if(iLen==SOCKET_ERROR) { int err=WSAGetLastError(); if (err==WSAECONNRESET ||err==WSAECONNABORTED) strtempMsg.Format(_T("%s连接中断"),p->Name); p->connectstate=false; return -2; //连接异常中断 } ...... .....省略代码 }while//do-while } //else closesocket(s); //关闭套接字 WSACleanup(); return 0; } 因为这个线程创建以后一直在运行,里面的循环体不停地从套接字中取数据并处理数据,但要是我所连接的服务端断开了我如何判断,并自动重新连接(而不用关程序重新开)。请高手在我代码基础上添加所需修改的地方,谢谢。

 

      你现在用的是阻塞模式的套接字。recv返回有几种情况,一种是收到数据返回,这时的返回值是>0的;一种是对方调用了close,这时recv会立即返回0;还有一种是返回-1(SOCKET_ERROR),这是由于网络出现了某种故障。如果出现网络故障,比如网络断了或者你拔了网线,大多数情况下recv不会立即返回,而是要等到套接字超时才返回-1(超时时间长度与具体系统有关,一般要几分钟)。

针对你的代码,要实现重连最简单的改进就是在函数里加一个循环,把你现在函数中全部代码放到这个循环内。

由于当网络出现故障时,recv要很长时间才能返回,如果你觉得这种方式不能接受,可以考虑select或者其他模型。

 

UINT CMonitorDlg::ThreadFunction(LPVOID pParam)
{
  WSADATA          wsaData;
  WSAStartup(...);

while(1)
{
  if (WaitSingleObject(hEventKill, 0) == WAIT_OBJECT_0)  // 这个hEventKill到主线程里创建,用于控制线程退出
  {
    break;
  }

  SOCKET          s;            //创建套接字
  int              port=PORT;
  int              iLen;                //从服务器接收的数据长度  
  TCHAR            recvbuf[BufferLen];  //接受数据的缓冲器

  struct sockaddr_in serv1;            //服务器端地址
  serv1.sin_family=AF_INET; //需要连接的服务器地址信息  
  serv1.sin_port=htons(port);  //需要连接的服务器地址信息
  serv1.sin_addr.s_addr=inet_addr(p->Ip);  //将命令行的IP地址转化为二进制表示的网络字节顺序IP地址
  s=socket(AF_INET,SOCK_STREAM,0);  

  if(s==INVALID_SOCKET)
  {
    AfxMessageBox("socket()failed");//套接字失败
    return 0;
  }


  if(connect(s,(struct sockaddr*)&serv1,sizeof(serv1))==INVALID_SOCKET)  
  {
    int b=WSAGetLastError();
    //连接失败
    }
  else    
  { //连接成功
  
      p->connectstate=true;
        

    do{

              在此循环体中我接收数据并对数据做处理(省略部分代码)
        iLen=recv(s,recvbuf,sizeof(recvbuf),0);    //接收套接字中的数据放入recvbuf1缓冲区

        if(iLen==0) // server调用了close
        {
          strtempMsg.Format(_T("局 %s 数据传送结束"),p->Name);
          break;
          //return 0;
        }
        else if(iLen==SOCKET_ERROR) // 网络错误
        {
          int err=WSAGetLastError();
          if (err==WSAECONNRESET ||err==WSAECONNABORTED)  
          strtempMsg.Format(_T("%s连接中断"),p->Name);

          p->connectstate=false;
        
        break;  // 退出do-while循环准备重连
        // return -2; //连接异常中断
                    }

          ......
          .....省略代码

    }while//do-while
  } //else
      
  closesocket(s); //关闭套接字
  } // end of while(1)

  WSACleanup();

return 0;

}

这么详细了,再不明白就不对了

 

原创粉丝点击