日本免费高清视频-国产福利视频导航-黄色在线播放国产-天天操天天操天天操天天操|www.shdianci.com

學(xué)無先后,達者為師

網(wǎng)站首頁 編程語言 正文

python進程結(jié)束后端口占用問題解析_python

作者:PinkD ? 更新時間: 2022-04-11 編程語言

socket分配

一個服務(wù)端進程向操作系統(tǒng)申請一個?scoket?來監(jiān)聽,但是當進程退出后,還未關(guān)閉的連接不會立即消失,而是會留給操作系統(tǒng)處理。操作系統(tǒng)會嘗試關(guān)閉這個連接。但是如果關(guān)閉時出現(xiàn)問題,這個連接就會一直處于?TIME_WAIT?或其他非正常狀態(tài),而這是相應(yīng)的端口還處于占用狀態(tài),如果這個時候再重新啟動這個服務(wù)端程序,就會出現(xiàn)地址被占用的情況

例子

測試代碼:

import socket
s = socket.socket()
s.bind(('0.0.0.0', 12345))
s.listen()
(client, addr) = s.accept()
print(client)
print(addr)

使用?nc?進行連接:

nc 127.0.0.1 12345

服務(wù)端會打印?client?和?addr?,然后正常退出,但是此時使用?netstat -anop | grep 12345?查看,發(fā)現(xiàn)對應(yīng)連接并沒有被立即釋放

tcp        0      0 127.0.0.1:12345         127.0.0.1:59408         TIME_WAIT   -                    timewait (28.18/0/0)

此時再次啟動服務(wù)端,發(fā)現(xiàn)報錯了:

Traceback (most recent call last):
  File "server.py", line 5, in <module>
    s.bind(('0.0.0.0', 12345))
OSError: [Errno 98] Address already in use

解決方案

使用?setsockopt?:

import socket
s = socket.socket()
s.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
s.bind(('0.0.0.0', 12345))
s.listen()
(client, addr) = s.accept()
print(client)
print(addr)

此時就不會出現(xiàn)地址被占用的提示了

在?c?中也有一樣的方法,只是方法聲明不同,?c?版的用法為

struct sockaddr_in addr;
addr.sin_family = AF_INET;
addr.sin_port = htons(12345);
addr.sin_addr.s_addr = htonl(INADDR_ANY);
int s = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
int reuse = 1;
setsockopt(s, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof(reuse));
bind(s, (struct sockaddr *) &addr, sizeof(addr))
listen(s, )
struct sockaddr_in in_addr;
int len = sizeof(in_addr);
int client = accept(socket, (struct sockaddr *) in_addr, &len);
//handle client
//...

其他

  • 發(fā)現(xiàn)除了?SO_REUSEADDR?之外還有一個?SO_REUSEPORT?的選項,查詢后得知是?BSD?獨有的,?Linux?并不能用
  • 如果是客戶端綁定端口用這個屬性可能會出現(xiàn)剛連接上服務(wù)器就莫名其妙收到一個?FIN?的問題,導(dǎo)致其立即關(guān)閉,因此客戶端使用此選項時需注意

參考鏈接

socket連接關(guān)閉問題分析

PortProtection

原文鏈接:https://blog.pinkd.moe/others/2018/07/29/TCP-reuse

欄目分類
最近更新