建立多个连接时如何在 C 中设置套接字超时?

2024-10-10 09:28:00
admin
原创
75
摘要:问题描述:我正在编写一个简单的程序,它与不同的服务器建立多个连接以进行状态检查。所有这些连接都是按需构建的;最多可以同时创建 10 个连接。我不喜欢每个套接字一个线程的想法,所以我将所有这些客户端套接字都设为非阻塞,并将它们放入 select() 池中。一切运行良好,直到我的客户抱怨说,当目标服务器停止响应时...

问题描述:

我正在编写一个简单的程序,它与不同的服务器建立多个连接以进行状态检查。所有这些连接都是按需构建的;最多可以同时创建 10 个连接。我不喜欢每个套接字一个线程的想法,所以我将所有这些客户端套接字都设为非阻塞,并将它们放入 select() 池中。

一切运行良好,直到我的客户抱怨说,当目标服务器停止响应时,他们等待的时间太长,才能够得到错误报告。

我查看了论坛中的几个主题。有人建议可以使用 alarm() 信号或在 select() 函数调用中设置超时。但我处理的是多个连接,而不是一个。当发生进程范围的超时信号时,我无法在所有其他连接中区分超时连接。

有没有办法改变系统默认的超时时间?


解决方案 1:

您可以使用 SO_RCVTIMEO 和 SO_SNDTIMEO 套接字选项来设置任何套接字操作的超时,如下所示:

    struct timeval timeout;      
    timeout.tv_sec = 10;
    timeout.tv_usec = 0;
    
    if (setsockopt (sockfd, SOL_SOCKET, SO_RCVTIMEO, &timeout,
                sizeof timeout) < 0)
        error("setsockopt failed
");

    if (setsockopt (sockfd, SOL_SOCKET, SO_SNDTIMEO, &timeout,
                sizeof timeout) < 0)
        error("setsockopt failed
");
    

编辑:来自setsockopt 手册页:

SO_SNDTIMEO是为输出操作设置超时值的选项。它接受一个 struct timeval 参数,该参数包含用于限制等待输出操作完成的秒数和微秒数。如果发送操作阻塞了这么长时间,它将返回部分计数,如果没有发送任何数据,则返回错误 EWOULDBLOCK。在当前实现中,每次向协议传送其他数据时都会重新启动此计时器,这意味着限制适用于大小从输出的低水位标记到高水位标记的输出部分。

SO_RCVTIMEO是一个用于设置输入操作超时值的选项。它接受一个 struct timeval 参数,该参数包含用于限制等待输入操作完成的时间(以秒和微秒为单位)。在当前实现中,每次协议收到其他数据时都会重新启动此计时器,因此该限制实际上是一个不活动计时器。如果接收操作被阻塞了这么长时间而没有收到其他数据,它将返回一个短计数,如果没有收到任何数据,则返回错误 EWOULDBLOCK。struct timeval 参数必须表示正时间间隔;否则,setsockopt() 将返回错误 EDOM。

解决方案 2:

我不确定我是否完全理解了这个问题,但猜测它与我遇到的问题有关,我正在使用 Qt 进行 TCP 套接字通信,全部是非阻塞的,无论是 Windows 还是 Linux。

希望在已连接的客户端发生故障或完全消失时快速收到通知,而不是等待默认的 900 多秒直到断开连接信号发出。实现此操作的技巧是将 SOL_TCP 层的 TCP_USER_TIMEOUT 套接字选项设置为所需值(以毫秒为单位)。

这是一个相对较新的选项,请参阅https://www.rfc-editor.org/rfc/rfc5482,但显然它运行良好,尝试使用 WinXP、Win7/x64 和 Kubuntu 12.04/x64,我选择的 10 秒结果有点长,但比我以前尝试过的任何其他方法都要好得多 ;-)

我遇到的唯一问题是找到正确的包含,因为显然这还没有添加到标准套接字包含中(但是..),所以最后我自己将它们定义如下:

#ifdef WIN32
    #include <winsock2.h>
#else
    #include <sys/socket.h>
#endif

#ifndef SOL_TCP
    #define SOL_TCP 6  // socket options TCP level
#endif
#ifndef TCP_USER_TIMEOUT
    #define TCP_USER_TIMEOUT 18  // how long for loss retry before timeout [ms]
#endif

设置此套接字选项仅当客户端已经连接时才有效,代码行如下:

int timeout = 10000;  // user timeout in milliseconds [ms]
setsockopt (fd, SOL_TCP, TCP_USER_TIMEOUT, (char*) &timeout, sizeof (timeout));

并且初始连接的失败被调用 connect() 时启动的计时器捕获,因为 Qt 没有信号,连接信号将不会被引发,因为没有连接,断开连接信号也不会被引发,因为还没有连接。

解决方案 3:

你不能实现自己的超时系统吗?

保留一个超时事件的排序列表,或者更好的是,按照 Heath 的建议,保留一个优先级堆。在 select 或 poll 调用中,使用超时列表顶部的超时值。当该超时到达时,执行与该超时相关的操作。

该操作可能是关闭尚未连接的套接字。

解决方案 4:

connect超时必须用非阻塞套接字来处理(GNU LibC文档)connect。您可以connect立即返回,然后使用select超时等待连接完成。

这里也解释了这一点:操作正在进行中,连接(函数)出现错误。

int wait_on_sock(int sock, long timeout, int r, int w)
{
    struct timeval tv = {0,0};
    fd_set fdset;
    fd_set *rfds, *wfds;
    int n, so_error;
    unsigned so_len;

    FD_ZERO (&fdset);
    FD_SET  (sock, &fdset);
    tv.tv_sec = timeout;
    tv.tv_usec = 0;

    TRACES ("wait in progress tv={%ld,%ld} ...
",
            tv.tv_sec, tv.tv_usec);

    if (r) rfds = &fdset; else rfds = NULL;
    if (w) wfds = &fdset; else wfds = NULL;

    TEMP_FAILURE_RETRY (n = select (sock+1, rfds, wfds, NULL, &tv));
    switch (n) {
    case 0:
        ERROR ("wait timed out
");
        return -errno;
    case -1:
        ERROR_SYS ("error during wait
");
        return -errno;
    default:
        // select tell us that sock is ready, test it
        so_len = sizeof(so_error);
        so_error = 0;
        getsockopt (sock, SOL_SOCKET, SO_ERROR, &so_error, &so_len);
        if (so_error == 0)
            return 0;
        errno = so_error;
        ERROR_SYS ("wait failed
");
        return -errno;
    }
}
相关推荐
  为什么项目管理通常仍然耗时且低效?您是否还在反复更新电子表格、淹没在便利贴中并参加每周更新会议?这确实是耗费时间和精力。借助软件工具的帮助,您可以一目了然地全面了解您的项目。如今,国内外有足够多优秀的项目管理软件可以帮助您掌控每个项目。什么是项目管理软件?项目管理软件是广泛行业用于项目规划、资源分配和调度的软件。它使项...
项目管理软件   601  
  华为IPD与传统研发模式的8大差异在快速变化的商业环境中,产品研发模式的选择直接决定了企业的市场响应速度和竞争力。华为作为全球领先的通信技术解决方案供应商,其成功在很大程度上得益于对产品研发模式的持续创新。华为引入并深度定制的集成产品开发(IPD)体系,相较于传统的研发模式,展现出了显著的差异和优势。本文将详细探讨华为...
IPD流程是谁发明的   7  
  如何通过IPD流程缩短产品上市时间?在快速变化的市场环境中,产品上市时间成为企业竞争力的关键因素之一。集成产品开发(IPD, Integrated Product Development)作为一种先进的产品研发管理方法,通过其结构化的流程设计和跨部门协作机制,显著缩短了产品上市时间,提高了市场响应速度。本文将深入探讨如...
华为IPD流程   9  
  在项目管理领域,IPD(Integrated Product Development,集成产品开发)流程图是连接创意、设计与市场成功的桥梁。它不仅是一个视觉工具,更是一种战略思维方式的体现,帮助团队高效协同,确保产品按时、按质、按量推向市场。尽管IPD流程图可能初看之下显得错综复杂,但只需掌握几个关键点,你便能轻松驾驭...
IPD开发流程管理   8  
  在项目管理领域,集成产品开发(IPD)流程被视为提升产品上市速度、增强团队协作与创新能力的重要工具。然而,尽管IPD流程拥有诸多优势,其实施过程中仍可能遭遇多种挑战,导致项目失败。本文旨在深入探讨八个常见的IPD流程失败原因,并提出相应的解决方法,以帮助项目管理者规避风险,确保项目成功。缺乏明确的项目目标与战略对齐IP...
IPD流程图   8  
热门文章
项目管理软件有哪些?
云禅道AD
禅道项目管理软件

云端的项目管理软件

尊享禅道项目软件收费版功能

无需维护,随时随地协同办公

内置subversion和git源码管理

每天备份,随时转为私有部署

免费试用