我正在尝试在服务器(在公共IP上)和客户端(在NAT上)之间使用UDP进行双向通信。我的逻辑说,如果服务器向IP发送一些数据,并且它从接收到数据包的端口,客户端应该仍然得到它,因为NAT将具有最终将数据包发送到客户端的映射。
客户端有2个进程,一个用于发送数据包,另一个用于接收数据。服务器继续等待数据,如果它获取数据,它会将数据发送回端口和IP,从而接收数据。
以下客户端代码:
client_recv.py
import socket
import sys
UDP_IP = '0.0.0.0'#my ip address in the local network
UDP_PORT = 5000
sock = socket.socket(socket.AF_INET, # Internet
socket.SOCK_DGRAM) # UDP
sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
sock.bind((UDP_IP, UDP_PORT))
while True:
data, addr = sock.recvfrom(1024)
print ("received message:" + data)
client_send.py
import socket
import time
import sys
UDP_IP = 'XXXXXXX.com'#external server IP address
UDP_PORT = 4000
MESSAGE = "Hi!"
sock = socket.socket(socket.AF_INET, # Internet
socket.SOCK_DGRAM) # UDP
sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
sock.bind(('0.0.0.0', 5000))
i = 0
while True:
sock.sendto(MESSAGE + ' # ' + str(i), (UDP_IP, UDP_PORT))
time.sleep(1)
i = i + 1
服务器(在XXXXX.com上 - 公共IP)获取数据包,但不获取客户端。以下是在服务器上接收和发送数据包的代码:
server_send_recv.py
import socket
sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
sock.bind(('0.0.0.0', 4000))
while True:
data, inetaddr = sock.recvfrom(1024)
print('Received ' + data)
(ip, port) = inetaddr
print("IP:"+str(ip)+",Port:"+str(port))
sock.sendto('Hi From Server (against ' + data + ')', (ip, port))
编辑1:
正如答案中已经说明的那样,必须使用相同的套接字,所以我做了以下工作,并且它有效。但是,我仍然不清楚为什么绑定到客户端上的同一端口的不同套接字(client_recv)将无法工作。由于重用相同的端口,client_recv应该正常工作吗?为什么它失败了,是什么让同一个套接字在客户端工作时发送和接收?
client_send_recv.py
import socket
import time
import sys
UDP_IP = 'twig-me.com'#external server IP address
UDP_PORT = 4000
MESSAGE = "Hi!"
sock = socket.socket(socket.AF_INET, # Internet
socket.SOCK_DGRAM) # UDP
sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
sock.bind(('0.0.0.0', 5000))
i = 0
while True:
print ("Sending message:" + str(i))
sock.sendto(MESSAGE + ' # ' + str(i), (UDP_IP, UDP_PORT))
time.sleep(1)
i = i + 1
data, addr = sock.recvfrom(1024)
print ("received message:" + data)
答案 0 :(得分:2)
TL; TR:问题与NAT无关。相反,它与...相比较不具体。套接字上更具体的绑定。
您在同一系统上有两个客户端:
UDP_IP = '0.0.0.0'#my ip address in the local network
sock.bind(('0.0.0.0', 5000))
sock.sendto(MESSAGE + ' # ' + str(i), (UDP_IP, UDP_PORT))
这意味着你最终得到两个插座:
如果数据包从服务器到达,它将转到最具体的套接字,即来自client_send.pl的套接字。因此client_recv.pl永远不会收到数据包。
如果您改为将client_recv.pl中的IP地址更改为本地系统的IP,则会获得两个套接字,这两个套接字都绑定到your-local-ip:5000
。由于在这种情况下没有最具体的套接字,因此在您的情况client_recv.pl
中,数据包实际上会被传送到第一个读取它的数据包。