python ZMQ recv() 超时后脚本未终止

python script not terminating after timeout in ZMQ recv()

到目前为止,这是我第一次使用 IPC,我编写了这个脚本:

#!/usr/bin/python

import zmq

context = zmq.Context()
socket = context.socket(zmq.PAIR)
socket.setsockopt(zmq.RCVTIMEO, 2000)
socket.connect ("ipc:///tmp/something")
socket.send(b"123")
try:
    message = socket.recv()
except:
    print("DEBUG!")
    message = None

当我的服务器脚本 运行(它只是发送一个答案)时一切正常。

但是当 .recv() 调用超时(例如因为没有服务器 运行 ),脚本不会在 "DEBUG!"-print,我必须使用 Ctrl+C.

手动停止它

我尝试断开并关闭套接字,但没有任何改变。

当我将整个脚本放入一个函数中并调用它时,在 KeyboardInterrupt:

上出现以下错误
^CException ignored in: <bound method Context.__del__ of <zmq.sugar.context.Context object at 0x7f16a36d5128>>
Traceback (most recent call last):
  File "/usr/lib/python3.6/site-packages/zmq/sugar/context.py", line 46, in __del__
    self.term()
  File "zmq/backend/cython/context.pyx", line 136, in zmq.backend.cython.context.Context.term (zmq/backend/cython/context.c:2339)
  File "zmq/backend/cython/checkrc.pxd", line 12, in zmq.backend.cython.checkrc._check_rc (zmq/backend/cython/context.c:3207)
KeyboardInterrupt

我是 运行 Python Arch 上的 PyZMQ 模块的 3.6.1 和版本 16.0.2 Linux。

可以将此作为标准的 ZeroMQ 基础设施设置策略:

LINGER 属性的默认值强制 socket 实例在尝试 .close() 时等待。因此,将其设置为 0 以避免在实例化时立即出现此功能/行为,而不是最终在终止时挂起。

import zmq
nIOthreads = 2                           # ____POLICY: set 2+: { 0: non-blocking, 1: blocking, 2: ...,  }
context = zmq.Context( nIOthreads )      # ____POLICY: set several IO-datapumps

socket  = context.socket( zmq.PAIR )
socket.setsockopt( zmq.LINGER,      0 )  # ____POLICY: set upon instantiations
socket.setsockopt( zmq.AFFINITY,    1 )  # ____POLICY: map upon IO-type thread
socket.setsockopt( zmq.RCVTIMEO, 2000 )

socket.connect( "ipc:///tmp/something" )
socket.send( b"123" )
try:
    message = socket.recv()
except:
    print( "DEBUG!" )
    message = None
finally:
    socket.close()                       # ____POLICY: graceful termination
    context.term()                       # ____POLICY: graceful termination

user3666197 的回答是正确的,因为 socket.setsockopt(zmq.LINGER, 0) 将解决您描述的问题。

但是根据 the docs,PAIR 套接字仅用于单个进程中的线程间通信。具体来说,它不会重新连接连接断开。

它可能通过 ipc 正常工作(我从未尝试过),但我建议使用不同的套接字类型。