【发布时间】:2012-09-20 14:37:50
【问题描述】:
我必须接收 UDP 广播(在 Ubuntu 中,如果这有什么不同的话)。使用 Wireshark,我可以看到从服务器机器发送的数据包,并且我可以看到它正在被我的客户端机器接收,但我的程序完全没有注意到。这就是我所拥有的:
sockaddr_in si_me, si_other;
int s;
assert((s=socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP))!=-1);
int port=6000;
int broadcast=1;
setsockopt(s, SOL_SOCKET, SO_BROADCAST,
&broadcast, sizeof broadcast);
memset(&si_me, 0, sizeof(si_me));
si_me.sin_family = AF_INET;
si_me.sin_port = htons(port);
si_me.sin_addr.s_addr = INADDR_ANY;
assert(::bind(s, (sockaddr *)&si_me, sizeof(sockaddr))!=-1);
while(1)
{
char buf[10000];
unsigned slen=sizeof(sockaddr);
recvfrom(s, buf, sizeof(buf)-1, 0, (sockaddr *)&si_other, &slen);
printf("recv: %s\n", buf);
}
它是在调试模式下编译的,编译过程中不会删除断言,我的程序只是阻塞在recvfrom。
为了接收非目标 UDP 广播,我是否必须跳过其他任何环节?
编辑:再多一点信息,我将两台计算机连接在一个专用交换机上,没有外部干扰。我的客户端计算机上还有第二个网卡,它连接到公司网络,它也可以工作。
我可以 ping 外部(互联网工作)和我的服务器机器(而且我可以在 Wireshark 中看到实际的数据包),但你永远不知道是什么导致了这个问题。
【问题讨论】:
-
这段代码适用于我(经过轻微修改以使其使用 C 编译器进行编译)。
bind可能绑定到错误的接口?netstat -an | grep 6000在客户端机器上返回什么? -
是的,它应该可以工作,但它不适合我。
netstat -an显示没有打开的 6000 端口,我已经仔细检查过(我也尝试使用“重用”udp 标志)。 -
bind()实际上并没有打开端口。只有connect()和listen()这样做。但是你不需要通过listen()打开一个端口来接收UDP广播。 -
@RemyLebeau 当然 bind() 打开一个端口。调用后查看 netstat 。另一方面,connect() 根本没有可见的效果。
-
由于 UDP 是无连接的,UDP 套接字上的
connect()仅在本地和远程 IP/端口对之间的套接字句柄内建立静态关联,仅此而已。当我提到connect()打开一个端口时,我指的是 TCP。在 UDP 套接字上调用bind()会将绑定的 IP/端口添加到 netstat 列表中,但它不会显示任何状态,这可能对所有 UDP 连接都是如此。