Python TCP服务器端口占用难题及解决方案
在使用Python编写TCP服务器时,程序退出后目标端口仍然被占用的情况时有发生,导致服务器无法立即重启。本文将分析此问题,并提供有效的。
问题:使用multiprocessing.pool创建进程池的TCP服务器,异常终止后,lsof -i :6001未显示端口6001被占用,但重启服务器时报错OSError: [Errno 98] Address already in use。netstat -anp | grep 6001显示大量TIME_WAIT状态连接。
原因分析:lsof显示的是进程正在使用的文件描述符,而TIME_WAIT状态的连接并非由任何进程直接持有。TCP连接关闭后,会将其保持在TIME_WAIT状态一段时间(通常60-120秒),以确保数据包可靠传输。在此期间,端口处于“占用”状态,但并非被进程使用,lsof无法检测到。netstat命令则清晰地显示了大量的TIME_WAIT连接,这正是问题根源。服务器意外退出导致客户端连接未正常关闭,从而阻塞端口。
解决方案:在服务器绑定端口前,设置套接字选项SO_REUSEADDR。这允许同一程序的多个套接字绑定同一端口。修改代码如下:
serversocket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) serversocket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) # 在bind之前添加 serversocket.bind(('0.0.0.0', port))
通过设置SO_REUSEADDR,可以有效避免TIME_WAIT状态导致的端口占用问题,从而实现服务器快速重启。 Linux 3.9及以上内核版本还提供SO_REUSEPORT选项,可实现更精细的端口复用,某些情况下可与SO_REUSEADDR结合使用。Windows系统可能需要额外设置SO_EXCLUSIVEADDRUSE选项。
以上就是TCP服务端程序退出后端口仍被占用是什么原因?的详细内容,更多请关注php中文网其它相关文章!