如何在python的socket recv方法上设置超时?

Tho*_*ler 100 python sockets timeout

我需要在python的socket recv方法上设置超时.怎么做?

Dan*_*ach 116

典型的方法是使用select()等待数据可用或直到超时发生.仅recv()在数据实际可用时调用.为了安全起见,我们还将套接字设置为非阻塞模式,以确保recv()永远不会无限期地阻塞. select()也可以用来一次等待多个套接字.

import select

mysocket.setblocking(0)

ready = select.select([mysocket], [], [], timeout_in_seconds)
if ready[0]:
    data = mysocket.recv(4096)
Run Code Online (Sandbox Code Playgroud)

如果你有很多打开文件描述符,poll()是一个更有效的替代方法select().

另一种选择是使用套接字上的所有操作设置超时socket.settimeout(),但我看到你在另一个答案中明确拒绝了该解决方案.

  • 使用`select`是好的,但你说"你不能"的部分是误导,因为有`socket.settimeout()`. (14认同)
  • 关于使用`select`的一个注意事项 - 如果你在Windows机器上运行,`select`依赖于WinSock库,它习惯在*某些*数据到达后立即返回,但不一定*全部*它的.因此,您需要合并一个循环来保持调用`select.select()`直到收到所有数据.你怎么知道你得到的所有数据(不幸的是)由你决定 - 它可能意味着寻找终结符字符串,一定数量的字节,或者只是等待定义的超时. (6认同)
  • 为什么有必要设置套接字没有非阻塞?我不认为这对select调用很重要(它会阻塞直到可以读取描述符或者在这种情况下超时到期)并且如果满足select,recv()将不会阻塞.我尝试使用recvfrom(),它似乎没有setblocking(0)正常工作. (3认同)

nos*_*klo 50

socket.settimeout()

  • 它没有超时recv(至少当我尝试它时).只有accept()超时. (14认同)
  • socket.recv()似乎在设置socket.settimeout()之后对我很有用,完全符合预期.我这样做了吗?其他人可以证实吗? (6认同)
  • 实际上,我可能会在select()返回时误解,所以请抓住之前的评论.Beej指南上面说的是落实科学的recv超时的有效方法:http://beej.us/guide/bgnet/output/html/singlepage/bgnet.html#indexId434909-198所以我相信是一个权威的来源. (4认同)
  • @Aeonaut我认为这次大部分时间都是recv(),但是有一个竞争条件.在socket.recv()中,Python(2.6)使用超时在内部调用select/poll,然后立即调用recv().因此,如果您使用阻塞套接字,并且在这两个调用之间另一个端点崩溃,您最终可能会无限期地挂在recv()上.如果你使用非阻塞套接字,python不会在内部调用select.select,所以我认为Daniel Stutzbach的答案是正确的. (3认同)
  • 我不确定为什么当这个解决方案是单行(更容易维护,实施错误的风险更小)并在引擎盖下使用 select(实现与@DanielStuzbach 答案相同)时,为什么首选使用 `select` 的解决方案. (2认同)

ube*_*kel 27

如前所述都select.select()socket.settimeout()正常工作.

请注意,您可能需要拨打settimeout两次以满足您的需求,例如

sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
sock.bind(("",0))
sock.listen(1)
# accept can throw socket.timeout
sock.settimeout(5.0)
conn, addr = sock.accept()

# recv can throw socket.timeout
conn.settimeout(5.0)
conn.recv(1024)
Run Code Online (Sandbox Code Playgroud)

  • 当你多次调用 `.settimeout()` 时,你可以首先调用 `setdefaulttimeout()` 方法。 (4认同)
  • 我认为无论你如何戳戳并推动这个挂起的功能,他都会遇到同样的事情.我现在尝试了2或4次超时,它仍然挂起.settimeout也挂起了. (3认同)

小智 12

您可以在收到响应之前设置超时,并在收到响应后将其设置回None:

sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

sock.settimeout(5.0)
data = sock.recv(1024)
sock.settimeout(None)
Run Code Online (Sandbox Code Playgroud)


小智 11

如果您实现服务器端,您正在寻找的超时是连接套接字的超时而不是主套接字的超时。换句话说,连接套接字对象还有另一个超时,它是socket.accept()方法的输出。所以:

sock.listen(1)
connection, client_address = sock.accept()
connection.settimeout(5)    # This is the one that affects recv() method.
connection.gettimeout()     # This should result 5
sock.gettimeout()           # This outputs None when not set previously, if I remember correctly.
Run Code Online (Sandbox Code Playgroud)

如果您实现客户端,那就很简单了。

sock.connect(server_address)
sock.settimeout(3)
Run Code Online (Sandbox Code Playgroud)


小智 6

试试这个它使用底层C。

timeval = struct.pack('ll', 2, 100)
s.setsockopt(socket.SOL_SOCKET, socket.SO_RCVTIMEO, timeval)
Run Code Online (Sandbox Code Playgroud)


Bri*_*eng 5

您可以使用socket.settimeout()which 接受表示秒数的整数参数。例如,socket.settimeout(1)将超时设置为 1 秒


Jos*_*ush 5

对顶级答案有些困惑,所以我写了一个小要点,并附有示例以便更好地理解。


选项1 - socket.settimeout()

如果sock.recv()等待时间超过定义的超时,则会引发异常。

import socket

sock = socket.create_connection(('neverssl.com', 80))
timeout_seconds = 2
sock.settimeout(timeout_seconds)
sock.send(b'GET / HTTP/1.1\r\nHost: neverssl.com\r\n\r\n')
data = sock.recv(4096)
data = sock.recv(4096) # <- will raise a socket.timeout exception here
Run Code Online (Sandbox Code Playgroud)

选项#2 - select.select()

等待数据发送,直到超时。我已经调整了丹尼尔的答案,所以它会引发一个例外

import select
import socket

def recv_timeout(sock, bytes_to_read, timeout_seconds):
    sock.setblocking(0)
    ready = select.select([sock], [], [], timeout_seconds)
    if ready[0]:
        return sock.recv(bytes_to_read)

    raise socket.timeout()

sock = socket.create_connection(('neverssl.com', 80))
timeout_seconds = 2
sock.send(b'GET / HTTP/1.1\r\nHost: neverssl.com\r\n\r\n')
data = recv_timeout(sock, 4096, timeout_seconds)
data = recv_timeout(sock, 4096, timeout_seconds) # <- will raise a socket.timeout exception here
Run Code Online (Sandbox Code Playgroud)