Answers:
典型的方法是使用select()等待数据可用或超时。仅recv()
在实际可用数据时调用。为了安全起见,我们还将套接字设置为非阻塞模式,以确保recv()
永远不会无限阻塞。 select()
也可以一次在多个插座上等待。
import select
mysocket.setblocking(0)
ready = select.select([mysocket], [], [], timeout_in_seconds)
if ready[0]:
data = mysocket.recv(4096)
如果您有很多打开的文件描述符,则poll()是更有效的替代方法select()
。
另一个选择是使用设置套接字上所有操作的超时socket.settimeout()
,但是我看到您在另一个答案中明确拒绝了该解决方案。
select
是好的,但是您说“不能”的部分会误导您,因为存在socket.settimeout()
。
select
-如果您在Windows计算机select
上运行,则依赖WinSock库,该库习惯于在某些数据到达(但不一定全部)后立即返回。因此,您需要合并一个循环以继续调用,select.select()
直到接收到所有数据为止。您知道如何获得所有数据(很不幸)取决于您-可能意味着寻找终止符字符串,一定数量的字节,或仅等待定义的超时。
ready[0]
只能是假的,如果有在服务器的响应没有身体?
select
当此解决方案是一个衬套(易于维护,实现错误的风险较小)并且在幕后使用select时首选使用的解决方案(实现与@DanielStuzbach答案相同)。
如前所述都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)
.settimeout()
多次调用时,您可以首先调用该setdefaulttimeout()
方法。
如果要实现服务器端,则要查找的超时是连接套接字的超时,而不是主套接字的超时。换句话说,连接套接字对象还有另一个超时,这是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.
如果实现客户端,那将很简单。
sock.connect(server_address)
sock.settimeout(3)
如先前的答复所述,您可以使用类似以下内容的.settimeout()
示例:
import socket
s = socket.socket()
s.settimeout(1) # Sets the socket to timeout after 1 second of no activity
host, port = "somehost", 4444
s.connect((host, port))
s.send("Hello World!\r\n")
try:
rec = s.recv(100) # try to receive 100 bytes
except socket.timeout: # fail after 1 second of no activity
print("Didn't receive data! [Timeout]")
finally:
s.close()
我希望这有帮助!!
尝试使用基础C。
timeval = struct.pack('ll', 2, 100)
s.setsockopt(socket.SOL_SOCKET, socket.SO_RCVTIMEO, timeval)
SO_RCVTIMEO
和设置发送和接收超时的不同值SO_SNDTIMEO
。
2
和为什么100
?哪个超时值?在什么单位?
timeval = struct.pack('ll', sec, usec)
s.setsockopt(socket.SOL_SOCKET, socket.SO_RCVTIMEO, timeval)
usec = 10000表示10毫秒
#! /usr/bin/python3.6
# -*- coding: utf-8 -*-
import socket
import time
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
s.setsockopt(socket.SOL_SOCKET, socket.SO_BROADCAST, 1)
s.settimeout(5)
PORT = 10801
s.bind(('', PORT))
print('Listening for broadcast at ', s.getsockname())
BUFFER_SIZE = 4096
while True:
try:
data, address = s.recvfrom(BUFFER_SIZE)
except socket.timeout:
print("Didn't receive data! [Timeout 5s]")
continue
喊到:https : //boltons.readthedocs.io/en/latest/socketutils.html
它提供了一个缓冲套接字,它提供了许多非常有用的功能,例如:
.recv_until() #recv until occurrence of bytes
.recv_closed() #recv until close
.peek() #peek at buffer but don't pop values
.settimeout() #configure timeout (including recv timeout)