关于connect函数超时太久的问题(不同系统的默认超时时长不一,但都很久)

解决方案思想:    通过把socket设置为非阻塞模式,然后通过select函数自己设置定时,检测套接字描述符是否可用。

windows端实现上代码: 过于粗略,仅呈思想

#include <winsock2.h>
#include <Windows.h>
#include <conio.h>
//#include <netdb.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <errno.h>
#include <malloc.h>
//#include <netinet/in.h>
//#include <arpa/inet.h>
//#include <sys/ioctl.h>
#include <stdarg.h>
//#include <fcntl.h>
#include <time.h>
int main(int argc, char *argv[]) 
{
    printf("==main===\n");fflush(stdout);
    //Winsows下启用socket
    WSADATA wsadata;
    if (WSAStartup(MAKEWORD(1, 1), &wsadata) == SOCKET_ERROR) {
        printf("WSAStartup() fail\n");fflush(stdout);
        exit(0);
    }
    printf("==WSAStartup ok ===\n");fflush(stdout);
      int sockClient = socket(AF_INET, SOCK_STREAM, 0);
      struct sockaddr_in addrSrv;
      addrSrv.sin_addr.s_addr = inet_addr(argv[1]);
      addrSrv.sin_family = AF_INET;
      addrSrv.sin_port = htons(atoi(argv[2]));
 //fcntl(sockClient, F_SETFL, fcntl(sockClient, F_GETFL, 0)|O_NONBLOCK); 
    printf("==goto  ioctlsocket ok ===\n");fflush(stdout);
    unsigned long ul=1;
     int ret=ioctlsocket(sockClient,FIONBIO,(unsigned long *)&ul);    //设置成非阻塞模式
    if(ret==SOCKET_ERROR)   //设置失败
    {
        printf("socket ioctl error%d\n", WSAGetLastError());fflush(stdout);
        return -10;
    }
    printf("==set sucess to connect===\n");fflush(stdout);
  int iRet = connect(sockClient, ( const struct sockaddr *)&addrSrv, sizeof(struct sockaddr_in));
 printf("connect iRet is %d, errmsg:%s\n", iRet, strerror(errno));
 fflush(stdout); // 返回-1不一定是异常
 if (iRet != 0) 
 {
/*     
   if(errno != EINPROGRESS)
 {
  printf("connect error:%s\n", strerror(errno)); fflush(stdout);
 }
 else
     */
 {
  printf("socket ===========\n");
  struct timeval tm = {5, 0}; 
  fd_set wset, rset; 
  FD_ZERO(&wset); 
  FD_ZERO(&rset); 
  FD_SET(sockClient, &wset); 
  FD_SET(sockClient, &rset); 
  int time1 = time(NULL);
  int n = select(sockClient + 1, &rset, &wset, NULL, &tm); 
  int time2 = time(NULL);
  printf("time gap is %d\n", time2 - time1);fflush(stdout);
  if(n < 0) 
  { 
   printf("select error, n is %d\n", n); fflush(stdout);
  } 
  else if(n == 0) 
  { 
   printf("connect time out\n"); fflush(stdout);
  } 
  else if (n == 1) 
  {
   if(FD_ISSET(sockClient, &wset)) 
   { 
    printf("connect ok!\n"); fflush(stdout);
    //fcntl(sockClient, F_SETFL, fcntl(sockClient, F_GETFL, 0) & ~O_NONBLOCK); 
   } 
   else
   { 
    printf("unknow error:%s\n", strerror(errno)); fflush(stdout);
   } 
  }
  else
  {
  printf("oh, not care now, n is %d\n", n);fflush(stdout);
  }
 } 
 } 
  printf("I am here!\n");fflush(stdout);
  //ioctlsocket(sockClient, FIONBIO, &ul); //设置为阻塞模式 阻塞时间为timeout时间
  getchar();
  closesocket(sockClient);
   
    WSACleanup();
    return 0;

}

 

 

linux端实现:

#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netdb.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <errno.h>
#include <malloc.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <sys/ioctl.h>
#include <stdarg.h>
#include <fcntl.h>
#include <time.h>
int main(int argc, char *argv[]) 
{
  int sockClient = socket(AF_INET, SOCK_STREAM, 0);
  struct sockaddr_in addrSrv;
  addrSrv.sin_addr.s_addr = inet_addr(argv[1]);
  addrSrv.sin_family = AF_INET;
  addrSrv.sin_port = htons(atoi(argv[2]));
  fcntl(sockClient, F_SETFL, fcntl(sockClient, F_GETFL, 0)|O_NONBLOCK); 
  int iRet = connect(sockClient, ( const struct sockaddr *)&addrSrv, sizeof(struct sockaddr_in));
  printf("connect iRet is %d, errmsg:%s\n", iRet, strerror(errno)); // 返回-1不一定是异常

 if (iRet != 0) 
 { 
   if(errno != EINPROGRESS)
   {
        printf("connect error:%s\n", strerror(errno)); 
   }
   else
   {
          struct timeval tm = {5, 0}; 
      fd_set wset, rset; 
      FD_ZERO(&wset); 
      FD_ZERO(&rset); 
      FD_SET(sockClient, &wset); 
      FD_SET(sockClient, &rset); 
      int time1 = time(NULL);
      int n = select(sockClient + 1, &rset, &wset, NULL, &tm); 
      int time2 = time(NULL);
      printf("time gap is %d\n", time2 - time1);
      if(n < 0) 
      { 
           printf("select error, n is %d\n", n); 
      } 
      else if(n == 0) 
      { 
           printf("connect time out\n"); 
      } 
      else if (n == 1) 
        {
         if(FD_ISSET(sockClient, &wset)) 
            { 
            printf("connect ok!\n"); 
            fcntl(sockClient, F_SETFL, fcntl(sockClient, F_GETFL, 0) & ~O_NONBLOCK); 
         } 
         else
         { 
            printf("unknow error:%s\n", strerror(errno)); 
          } 
      }
       else
         {
           printf("oh, not care now, n is %d\n", n);
      }
    } 
  } 
  printf("I am here!\n");
  getchar();
  close(sockClient);
  return 0;
}
 

 

http://cache.baiducontent.com/c?m=YG5_oMGpXgIPmbhjnGo8iuOkg6r5BG-IOYPiHxsHndO7uml0F8mDauTl7Kgmpr0Hw3IFclRUo4wjnRZZTypvCOJrTUVpllsN8-It0LlsonT5YhmtpZpnciKQ0rrQlEtLEk4J2I7qmec2SC0QyazL1q&p=cb769a47898c13ff57ec966d5b40&newp=9c6fd10586cc40ad01a2c7710f5e92695d0fc20e3bd3da01298ffe0cc4241a1a1a3aecbf2c251b02d9c277620aad4e5ceff132743d0034f1f689df08d2ecce7e76c7&s=45c48cce2e2d7fbd&user=baidu&fm=sc&query=windows+connect&qid=89bd31ba0043d24b&p1=2

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

字正腔圆

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值