Linux下connect函数 阻塞 与 非阻塞 问题

简介: Linux下connect函数 阻塞 与 非阻塞 问题

一、概述

linux系统下,connect函数是阻塞的,阻塞时间的长度与系统相关。而如果把套接字设置成非阻塞,调用connect函数时会报错Operation now in progress,且errno被设置为EINPROGRESS。下面将分析非阻塞时调用connect报错的原因,以及提供两个方法解决connet函数阻塞太久的问题。


二、connect函数报Operation now in progress的原因

connect函数报错Operation now in progress,且errno被设置为EINPROGRESS的原因是套接字被设置为非阻塞了。建立TCP连接会涉及到三次握手的过程,connect函数会一直等到收到自己的SYN的ACK为止,所以会阻塞一段时间;如果套接字设置成非阻塞,connect函数会立即返回,但此时已经发起的TCP三次握手仍在进行,所以connect会返回一个EINPROGRESS错误,表示操作正在进行中.


三、解决connet函数阻塞太久的问题

三次握手的过程就决定了connect函数是需要阻塞一段时间的,而我们是向避免它阻塞太久了,影响程序执行。下面提供2个可行的方法,仅供参考。

3.1 设置成非阻塞,且使用select等到连接的建立

// 接服务器并返回套接字,timeoutMs 毫秒
int tcpClientConnect(unsigned int u32SerIp, unsigned short port, int timeoutMs)
{
  // 1、创建tcp套接字
  int sockFd = -1;
  if ((sockFd = socket(PF_INET, SOCK_STREAM, 0)) == -1) 
  { 
    printf("[%s %d] socket failed\n",__FILE__,__LINE__); 
    return -1;
  } 
  // 2、socket 设置非阻塞
  int flags = fcntl(sockFd,F_GETFL,0);
  if(-1 == fcntl(sockFd,F_SETFL,flags|O_NONBLOCK))
  {
    printf("Set socket unblock failed!");
  }
  // 3、准备服务器地址信息、连接
  struct sockaddr_in server_addr;; 
  server_addr.sin_family=PF_INET; 
  server_addr.sin_port=htons(port); 
  server_addr.sin_addr.s_addr = u32SerIp; 
  bzero(&(server_addr.sin_zero), 0); 
    if (connect(sockFd, (struct sockaddr *)&server_addr, sizeof(struct sockaddr)) < 0)
    {
    if(errno != EINPROGRESS)
    {
      printf("[%s %d] Connect fail!\n",__FILE__,__LINE__);
      close(sockFd);
          return -1;
    }
    }
  // 4、select等待
  fd_set rset,wset;
  FD_ZERO(&rset);
  FD_SET(sockFd,&rset);
  wset = rset;
  struct timeval tv;
  tv.tv_sec = timeoutMs/1000;
  tv.tv_usec = (timeoutMs%1000)*1000;
  if(select(sockFd+1, &rset, &wset, NULL, &tv)==0)
  {
    printf("[%s %d] select fail!\n",__FILE__,__LINE__);
    close(sockFd);
      return -1;    
  }
  if(FD_ISSET(sockFd,&rset) || FD_ISSET(sockFd,&wset))
  {
    int error;
    socklen_t len = sizeof(error);
    if(getsockopt(sockFd, SOL_SOCKET, SO_ERROR, &error, &len)<0)
      return -1;  
  }
  printf("[%s %d] Connect pass! %d\n",__FILE__,__LINE__,sockFd);
  return sockFd;
}

3.2 使用套接字选项设置发送超时

// timeoutMs 毫秒
int tcpClientConnect(uint32_t u32SerIp, uint16_t port, int timeoutMs)//连接服务器并放回套接字 2022-06-23 20:02:34
{
  // 1、创建tcp套接字
  int sockFd = -1;
  if ((sockFd = socket(PF_INET, SOCK_STREAM, 0)) == -1) 
  { 
    printf("[%s %d] socket failed\n",__FILE__,__LINE__); 
    return -2;
  } 
  // 2、设置 sockFd 超时时间*/
  if(timeoutMs > 0)
  {
      struct timeval tv;
    tv.tv_sec = timeoutMs/1000;
    tv.tv_usec = (timeoutMs%1000)*1000;
    setsockopt(sockFd, SOL_SOCKET, SO_SNDTIMEO, &tv, sizeof(tv));
  }
  // 3、准备服务器地址信息、连接
  struct sockaddr_in server_addr;; 
  server_addr.sin_family=PF_INET; 
  server_addr.sin_port=htons(port); 
  server_addr.sin_addr.s_addr = u32SerIp; 
  bzero(&(server_addr.sin_zero), 0); 
    if (connect(sockFd, (struct sockaddr *)&server_addr, sizeof(struct sockaddr)) < 0)
    {
    printf("[%s %d] Connect fail!\n",__FILE__,__LINE__);
    close(sockFd);
        return -1;
    }
  printf("[%s %d] Connect pass! %d\n",__FILE__,__LINE__,sockFd);
  return sockFd;
}


目录
相关文章
|
6天前
|
Linux
【Linux】System V信号量详解以及semget()、semctl()和semop()函数讲解
System V信号量的概念及其在Linux中的使用,包括 `semget()`、`semctl()`和 `semop()`函数的具体使用方法。通过实际代码示例,演示了如何创建、初始化和使用信号量进行进程间同步。掌握这些知识,可以有效解决多进程编程中的同步问题,提高程序的可靠性和稳定性。
45 19
|
8天前
|
Linux Android开发 开发者
linux m、mm、mmm函数和make的区别
通过理解和合理使用这些命令,可以更高效地进行项目构建和管理,特别是在复杂的 Android 开发环境中。
39 18
|
16天前
|
存储 监控 Linux
嵌入式Linux系统编程 — 5.3 times、clock函数获取进程时间
在嵌入式Linux系统编程中,`times`和 `clock`函数是获取进程时间的两个重要工具。`times`函数提供了更详细的进程和子进程时间信息,而 `clock`函数则提供了更简单的处理器时间获取方法。根据具体需求选择合适的函数,可以更有效地进行性能分析和资源管理。通过本文的介绍,希望能帮助您更好地理解和使用这两个函数,提高嵌入式系统编程的效率和效果。
79 13
|
4月前
|
Linux Shell
Linux系统编程:掌握popen函数的使用
记得在使用完 `popen`打开的流后,总是使用 `pclose`来正确关闭它,并回收资源。这种做法符合良好的编程习惯,有助于保持程序的健壮性和稳定性。
168 6
|
4月前
|
Linux Shell
Linux系统编程:掌握popen函数的使用
记得在使用完 `popen`打开的流后,总是使用 `pclose`来正确关闭它,并回收资源。这种做法符合良好的编程习惯,有助于保持程序的健壮性和稳定性。
188 3
|
4月前
|
Linux
在Linux内核中根据函数指针输出函数名称
在Linux内核中根据函数指针输出函数名称
|
5月前
|
Linux PHP
Linux CentOS 宝塔 Suhosin禁用php5.6版本eval函数详细图文教程
【8月更文挑战第27天】本文介绍两种禁用PHP执行的方法:使用`PHP_diseval_extension`禁用和通过`suhosin`禁用。由于`suhosin`不支持PHP8,仅适用于PHP7及以下版本,若服务器安装了PHP5.6,则需对应安装`suhosin-0.9.38`版本。文章提供了详细的安装步骤,并强调了宝塔环境下与普通环境下的PHP路径差异。安装完成后,在`php.ini`中添加`suhosin.so`扩展并设置`executor.disable_eval = on`以禁用执行功能。最后通过测试代码验证是否成功禁用,并重启`php-fpm`服务生效。
66 2
|
5月前
|
Shell Linux C语言
Linux0.11 execve函数(六)
Linux0.11 execve函数(六)
93 1
|
5月前
|
Linux
Linux0.11 文件打开open函数(五)
Linux0.11 文件打开open函数(五)
58 0
|
5月前
|
存储 Linux 调度
Linux 0.11 fork 函数(二)
Linux 0.11 fork 函数(二)
54 0