UDP 套接字是可以使用 connect 系統調用連接到指定的地址的。從此以后,這個套接字只會接收來自這個地址的數據,而且可以使用 send 系統調用直接發數據而不用指定地址。可以再次調用 connect 來連接到別的地方。但是在 Python 里,一旦調用 connect 之后,就再也回不到最初的能夠接收從任意地址來的數據的狀態了!
這是 Python 的 API 限制,沒辦法給 connect 方法傳遞到 AF_UNSPEC 地址簇(在 C 代碼里寫死了的)。C 里邊就可以做到的(代碼來自這里):
代碼如下:
int disconnect_udp_sock(int fd) {
struct sockaddr_in sin;
memset((char *)&sin, 0, sizeof(sin));
sin.sin_family = AF_UNSPEC;
return (connect(fd, (struct sockaddr *)&sin, sizeof(sin)));
}
不過既然是 Python 的限制,拿 ctypes 就可以繞過了嘛,有些麻煩就是了:
代碼如下:
from ctypes import CDLL, create_string_buffer
def disconnect(sock):
libc = CDLL("libc.so.6")
buf = create_string_buffer(16) # sizeof struct sockaddr_in
libc.connect(sock.fileno(), buf, 16)
AF_UNSPEC 的值是 0,所以把一個和 struct sockaddr_in 一樣長的全零緩沖區傳給 connect 就可以了 :-)
新聞熱點
疑難解答