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

2024-10-10 09:28:00
admin
原创
188
摘要:问题描述:我正在编写一个简单的程序,它与不同的服务器建立多个连接以进行状态检查。所有这些连接都是按需构建的;最多可以同时创建 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;
    }
}
相关推荐
  为什么项目管理通常仍然耗时且低效?您是否还在反复更新电子表格、淹没在便利贴中并参加每周更新会议?这确实是耗费时间和精力。借助软件工具的帮助,您可以一目了然地全面了解您的项目。如今,国内外有足够多优秀的项目管理软件可以帮助您掌控每个项目。什么是项目管理软件?项目管理软件是广泛行业用于项目规划、资源分配和调度的软件。它使项...
项目管理软件   1129  
  IPD(Integrated Product Development,集成产品开发)流程是一种广泛应用于高科技和制造业的产品开发方法论。它通过跨职能团队的紧密协作,将产品开发周期缩短,同时提高产品质量和市场成功率。在IPD流程中,CDCP(Concept Decision Checkpoint,概念决策检查点)是一个关...
IPD培训课程   87  
  研发IPD(集成产品开发)流程作为一种系统化的产品开发方法,已经在许多行业中得到广泛应用。它不仅能够提升产品开发的效率和质量,还能够通过优化流程和资源分配,显著提高客户满意度。客户满意度是企业长期成功的关键因素之一,而IPD流程通过其独特的结构和机制,能够确保产品从概念到市场交付的每个环节都围绕客户需求展开。本文将深入...
IPD流程   79  
  IPD(Integrated Product Development,集成产品开发)流程是一种以跨职能团队协作为核心的产品开发方法,旨在通过优化资源分配、提高沟通效率以及减少返工,从而缩短项目周期并提升产品质量。随着企业对产品上市速度的要求越来越高,IPD流程的应用价值愈发凸显。通过整合产品开发过程中的各个环节,IPD...
IPD项目管理咨询   89  
  跨部门沟通是企业运营中不可或缺的一环,尤其在复杂的产品开发过程中,不同部门之间的协作效率直接影响项目的成败。集成产品开发(IPD)作为一种系统化的项目管理方法,旨在通过优化流程和增强团队协作来提升产品开发的效率和质量。然而,跨部门沟通的复杂性往往成为IPD实施中的一大挑战。部门之间的目标差异、信息不对称以及沟通渠道不畅...
IPD是什么意思   82  
热门文章
项目管理软件有哪些?
云禅道AD
禅道项目管理软件

云端的项目管理软件

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

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

内置subversion和git源码管理

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

免费试用