Python:非阻塞套接字或Asynchronos I / O.

时间:2011-06-16 15:46:14

标签: python sockets nonblocking

我是Python新手,目前必须编写一个python套接字作为脚本运行,通过TCP / IP(气象站)与设备通信。
设备充当服务器端(通过IP侦听:PORT,接受连接,接收请求,传输数据) 我只需要发送一条消息,接收答案,然后安静地关闭并关闭套接字。

try:
    comSocket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
except socket.error, msg:
    sys.stderr.write("[ERROR] %s\n" % msg[1])
    sys.exit(1)

try:
   comSocket.connect((''))
except socket.error, msg:
   sys.stderr.write("[ERROR] %s\n" % msg[1])
   sys.exit(2)

comSocket.send('\r')
comSocket.recv(128)
comSocket.send('\r')
comSocket.recv(128)
comSocket.send('\r\r')
comSocket.recv(128)

comSocket.send('1I\r\r3I\r\r4I\r\r13I\r\r5I\r\r8I\r\r7I\r\r9I\r\r')

rawData = comSocket.recv(512)

comSocket.shutdown(1)
comSocket.close()

我遇到的问题是:
通信信道不可靠,设备很慢。因此,有时设备响应长度为0的消息(只是一个ACK),我的代码将冻结并等待永远的响应 这段代码包含涉及SOCKET的部分,整个代码将在CRON下运行,因此冻结不是理想的行为。

我的问题是:
Python中处理该行为的最佳方式是什么,以便代码不会冻结并永远等待,但会尝试继续下一个发送(或者这样)。

3 个答案:

答案 0 :(得分:11)

您可以尝试使用超时方法,例如Russel代码,也可以使用非阻塞套接字,如下面的代码所示。它永远不会在socket.recv处阻塞,您可以在循环内使用它来重试您想要的次数。这样你的程序就不会在超时时挂起。这样,您可以测试数据是否可用,如果没有,您可以执行其他操作,稍后再试。

socket.setblocking(0)
while (retry_condition):
    try:
        data = socket.recv(512)
    except socket.error:
        '''no data yet..'''

答案 1 :(得分:3)

我建议使用eventlet和绿色线程。

Twisted是一个很好的库,但对于这样一个简单的用例,它有一点陡峭的学习曲线。

查看一些示例here

答案 2 :(得分:0)

在接收之前尝试在套接字上设置超时:

comSocket.settimeout(5.0)
try:
    rawData = comSocket.recv(512)
except socket.timeout:
    print "No response from server"