【问题标题】:Socket waiting for connection timeout套接字等待连接超时
【发布时间】:2015-11-05 16:33:59
【问题描述】:

我正在尝试实现一个超时,当在定义的时间间隔内没有接收到连接时终止 python 脚本。到目前为止,我设法使用以下代码实现超时:

import sys
import socket

# Create a TCP/IP socket
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

# Bind the socket to the port
server_address = ('192.168.43.112', 5001)
print >>sys.stderr, 'starting up on %s port %s' % server_address
sock.bind(server_address)

# Listen for incoming connections
sock.listen(1)

while True:
    try:
        # Wait for a connection
        print >>sys.stderr, 'waiting for a connection'
        connection, client_address = sock.accept()

        try:
            print >>sys.stderr, 'connection from', client_address

            # Receive the data in small chunks and retransmit it
            while True:
                data = connection.recv(16)
                print >>sys.stderr, 'received "%s"' % data
                if data:
                    print >>sys.stderr, 'Do stuff here'
                else:
                    print >>sys.stderr, 'no more data from', client_address
                    sock.settimeout(5)
                    break

        finally:
            # Clean up the connection
            connection.close()

    except socket.timeout:
        break

在建立连接并结束同一连接后,代码正常工作,5 秒后脚本终止。但是,如果在超时窗口期间我尝试建立另一个连接,则会出现以下错误:

starting up on 192.168.43.112 port 5001
waiting for a connection
connection from ('192.168.43.1', 47550)
received "Data 0
"
Do stuff here
received ""
no more data from ('192.168.43.1', 47550)
waiting for a connection
connection from ('192.168.43.1', 39010)
---------------------------------------------------------------------------
error                                     Traceback (most recent call last)
/Users/location/Desktop/sandbox/data_fetcher.py in <module>()
     24             # Receive the data in small chunks and retransmit it
     25             while True:
---> 26                 data = connection.recv(16)
     27                 print >>sys.stderr, 'received "%s"' % data
     28                 if data:

error: [Errno 35] Resource temporarily unavailable

【问题讨论】:

    标签: python sockets timeout


    【解决方案1】:

    我不完全确定你希望这一切如何工作,我发现它现在以这种方式发生有点令人惊讶(我没想到超时会产生这种效果),但基于 EAGAIN错误(errno 35),正在发生的事情是主套接字上的超时(仅在您拥有第一个连接后才设置)导致 second 接受的套接字处于非阻塞状态模式也是如此。这意味着,当您调用 connection.recv 并且立即没有数据时,您会得到 OSError 的提升。

    我怀疑其中一些在操作系统之间可能会有所不同,但我能够在 FreeBSD 上重现这一点(您可能在 Linux 上运行)。

    一个可以解决这个问题的最小更改——我认为这不一定是最好的编码方式,但它确实有效——是显式地将接受的套接字设置为阻塞:

            # Wait for a connection
            print >>sys.stderr, 'waiting for a connection'
            connection, client_address = sock.accept()
    
            connection.setblocking(1)
    

    这样,代码表现得更好(我添加了一个小型测试框架,将您的代码作为一个单独的进程分离出来,然后以不同的延迟建立多个连接)。

    【讨论】:

    • 这确实成功了。我在 OS X 上运行它。也感谢您的解释。
    猜你喜欢
    • 2011-03-06
    • 2021-05-23
    • 1970-01-01
    • 2017-01-17
    • 2011-05-11
    • 2017-03-08
    • 2017-12-10
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多