如何关闭被终止程序留下的开放套接字?
我有一个Python应用程序,它会打开一个简单的TCP套接字,用来和另一台主机上的Python应用程序进行通信。有时候,这个程序会出错,或者我直接把它关闭,这样一来,套接字可能会在一段时间内保持打开状态。
下次我运行这个程序时,就会出现这个错误:
socket.error: [Errno 98] Address already in use
因为这个程序总是尝试使用同一个端口,所以看起来好像它仍然是打开的。我检查过,确定程序没有在后台运行,但我的地址仍然被占用。
那么,我该如何手动(或者其他方式)关闭一个套接字/地址,以便我的程序可以立即重新使用它呢?
更新
根据Mike的回答,我查看了socket(7)
页面,并了解了SO_REUSEADDR:
SO_REUSEADDR Indicates that the rules used in validating addresses supplied in a bind(2) call should allow reuse of local addresses. For AF_INET sockets this means that a socket may bind, except when there is an active listening socket bound to the address. When the listen‐ ing socket is bound to INADDR_ANY with a specific port then it is not possible to bind to this port for any local address. Argument is an integer boolean flag.
3 个回答
5
你可能搞混了套接字、连接和端口。套接字是连接的端点,而连接实际上是由五个部分组成的:{协议、本地IP、本地端口、远程IP、远程端口}。被关闭的程序的套接字已经被操作系统关闭,连接也是如此。连接唯一留下的痕迹就是对方的套接字和对方主机上的相应端口。所以你真正应该问的是如何重新使用本地的端口。 对于这个问题,答案就是使用SO_REUSEADDR,正如其他回答所提到的那样。
12
你可以试试用Twisted来处理网络连接。Mike提到的低级别选项SO_REUSEADDR
是对的,但他没说在Windows上设置这个选项并不是个好主意。这种事情Twisted会自动帮你处理。使用socket模块时,你需要关注很多这种繁琐的低级细节,但如果你用像Twisted这样的高级库,就可以不用担心这些了。
59
假设你的套接字叫做 s
... 在把它绑定到一个接口之前,你需要在服务器的套接字上设置 socket.SO_REUSEADDR
... 这样做可以让你在 TCP 服务器崩溃后立即重启它...
s.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
s.bind((ADDR, PORT))