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

2024-10-10 09:28:00
admin
原创
219
摘要:问题描述:我正在编写一个简单的程序,它与不同的服务器建立多个连接以进行状态检查。所有这些连接都是按需构建的;最多可以同时创建 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;
    }
}
相关推荐
  为什么项目管理通常仍然耗时且低效?您是否还在反复更新电子表格、淹没在便利贴中并参加每周更新会议?这确实是耗费时间和精力。借助软件工具的帮助,您可以一目了然地全面了解您的项目。如今,国内外有足够多优秀的项目管理软件可以帮助您掌控每个项目。什么是项目管理软件?项目管理软件是广泛行业用于项目规划、资源分配和调度的软件。它使项...
项目管理软件   1344  
  信创产业的蓬勃发展推动着各行业数字化转型加速,数据库迁移作为其中关键一环,面临诸多挑战。信创数据库迁移旨在将传统数据库平稳过渡到信创环境,以满足自主可控、安全可靠的需求。这一过程涉及技术、业务等多方面因素,稍有不慎就可能出现各种问题,影响业务的正常运行。深入探讨信创数据库迁移过程中的常见问题及解决方案,对于保障迁移工作...
2027年信创国产化   54  
  随着信息技术的飞速发展,信创国产化成为了国家战略的重要组成部分。国产化信创产品名录涵盖了众多领域,其在各个关键应用场景中发挥着重要作用。而信创国产化操作系统作为其中的核心环节,具备五大核心优势,为我国信息技术产业的自主可控发展提供了坚实支撑。关键应用场景之办公领域在办公领域,国产化信创产品有着广泛且深入的应用。如今,越...
国产信创系统   50  
  随着信息技术的飞速发展,信创国产化操作系统在政府部门的推广应用具有重要的战略意义。它不仅关乎国家信息安全,更是推动国内信息技术产业自主创新、实现科技自立自强的关键举措。在当前复杂的国际形势下,政府部门积极推广信创国产化操作系统,对于保障国家政务信息的安全稳定运行,提升信息技术的自主可控能力,具有不可替代的重要作用。推广...
信创产品有哪些   37  
  在企业数字化转型的进程中,信创数据库解决方案的选择至关重要。它不仅关乎企业数据的安全存储与管理,更影响着企业业务的稳定运行与未来发展。合适的信创数据库能够助力企业在复杂多变的市场环境中提升竞争力,保障数据主权与安全。然而,面对市场上众多的信创数据库产品和解决方案,企业往往感到困惑,不知如何做出正确的选择。接下来,我们将...
信创电脑   29  
热门文章
项目管理软件有哪些?
云禅道AD
禅道项目管理软件

云端的项目管理软件

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

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

内置subversion和git源码管理

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

免费试用