我有一个Python协议在CentOS 6.4/64位下运行。其中我有TCP服务器端口7007。在某些情况下,如更新新版本或维护或在运行中重新启动以刷新缓冲区,我需要重新启动应用程序:
服务器.py:
class AServer(threading.Thread):
def __init__(self, port):
threading.Thread.__init__(self)
self.port = port
def run(self):
host = ''
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.bind((host, self.port))
print bgcolors.BOOT
s.listen(1)
conn, addr = s.accept()
print bgcolors.OK + 'contact', addr, 'on', self.now()
while 1:
try:
data = conn.recv(1024)
except socket.error:
print bgcolors.OK + 'lost', addr, 'waiting..'
s.listen(1)
conn, addr = s.accept()
print bgcolors.OK + 'contact', addr, 'on', self.now()
continue
if not data:
.....
...
t = AServer(7007)
t.start()
立即紧急重新启动(预期在1秒内执行),但失败:
$ ps aux | awk '/server.py/ {print $2}' | head -1 | xargs kill -9;
$ nohup python /var/tmp/py-protocol/server.py &
[root@IPSecVPN protocol]# python server.py
Exception in thread Thread-1:
Traceback (most recent call last):
File "/usr/lib64/python2.6/threading.py", line 532, in __bootstrap_inner
self.run()
File "server.py", line 236, in run
s.bind((host, self.port))
File "<string>", line 1, in bind
error: [Errno 98] Address already in use
3条答案
按热度按时间ktecyv1j1#
你的套接字处于TIME_WAIT状态,这就是为什么即使你的程序已经退出,地址仍然在使用的原因。你可以在套接字上设置SO_REUSEADDR,以便在套接字退出TIME_WAIT状态之前重用它。Python文档建议如下:
dddzy1tm2#
TCP有一个TIMEWAIT计时器,它可以让连接保持一段时间(在大多数操作系统上大约为2分钟)。所以,如果你已经绑定并连接了一个端口,那么关闭它可能会使它处于TIMEWAIT状态。更准确地说,只有TCP连接的一端处于TIMEWAIT状态。
这是一个很好的讨论TIMEWAIT:What is the cost of many TIME_WAIT on the server side?
同意@clj解决方案,设置SO_REUSEADDR是一个好主意(+1)。
sdnqo3pr3#
打开终端:
列出所有正在运行的服务并找到要终止的ID。