gpt4 book ai didi

Python 2.7 socket.timeout 行为

转载 作者:太空狗 更新时间:2023-10-30 02:58:37 25 4
gpt4 key购买 nike

下面是两个简单的 python 函数。首先尝试连接到 666 上的 test.com 域(主机名有效但端口无效)。第二次尝试连接到端口 993 上的 imap.mail.outlook.com(主机名有效,但看起来不供公众使用/访问)。

def fn_outlook(timeout):
try:
socket.create_connection(("imap.mail.outlook.com", 993), timeout=timeout)
except socket.timeout:
pass


def fn_test(timeout):
try:
socket.create_connection(("test.com", 666), timeout=timeout)
except socket.timeout:
pass

下面是具有不同超时的函数的执行时间:

In [14]: %time fn_test(1)
CPU times: user 644 µs, sys: 1.07 ms, total: 1.71 ms
Wall time: 1 s

In [15]: %time fn_test(2)
CPU times: user 589 µs, sys: 1.15 ms, total: 1.74 ms
Wall time: 2 s

In [16]: %time fn_outlook(2)
CPU times: user 838 µs, sys: 2.24 ms, total: 3.08 ms
Wall time: 7.15 s

In [17]: %time fn_outlook(4)
CPU times: user 705 µs, sys: 1.18 ms, total: 1.88 ms
Wall time: 12 s

In [18]: %time fn_test(4)
CPU times: user 483 µs, sys: 795 µs, total: 1.28 ms
Wall time: 4.42 s

对于 test.com 连接将在 timeout 参数中指定的〜相同时间后超时。但是对于 imap.mail.outlook.com 事情变得有趣了——套接字连接忽略超时参数。准确地说 - 不是忽略,而是总是在一段较长的时间后超时连接。

我可能认为此行为源自 imap.mail.outlook.com 服务器,而不是来自套接字模块。

最佳答案

首先,您可以将您的功能统一为:

def fn_connect(host, port, timeout):
try:
s = socket.create_connection((host, port), timeout=timeout)
except socket.timeout:
return None
else:
return s

并这样称呼他们:

IMAP_HOST = "imap.mail.outlook.com"
IMAP_PORT = 993
TEST_HOST = "test.com"
TEST_PORT = 666
s1 = fn_connect(IMAP_HOST, IMAP_PORT, 2)
s2 = fn_connect(TEST_HOST, TEST_PORT, 1)
#and so on....

我返回套接字以便之后能够正确关闭它(如果 not None)。

问题在于底层套接字机制如何解析主机名; create_connection 调用 getaddrinfo ,对于返回的每个地址,它会尝试创建一个套接字并连接到它(并且每个套接字都有您指定的超时)。因此,您的 2 个地址的结果:

  • TEST_HOSTTEST_PORT

    socket.getaddrinfo("test.com", 666)
    [(2, 0, 0, '', ('69.172.200.235', 666))]
  • IMAP_HOST, IMAP_PORT

    socket.getaddrinfo("imap.mail.outlook.com", 993)
    [(2, 0, 0, '', ('207.46.163.247', 993)),
    (2, 0, 0, '', ('207.46.163.138', 993)),
    (2, 0, 0, '', ('207.46.163.215', 993))]

如您所见,对于 IMAP_HOSTIMAP_PORT 它返回 3 个单独的地址(对于 TEST_HOSTTEST_PORT> 它只返回一个)。由于您指定它们都不起作用,因此它将尝试连接到所有这些,导致一般超时比您指定的大 ~3 倍。

关于Python 2.7 socket.timeout 行为,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33503585/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com