首页 > 解决方案 > 无法接收 UDP 广播消息

问题描述

我正在尝试组装一个 UDP 监视器。我认为这将是直截了当的,但我所尝试的一切都失败了。

目前,我正在使用 Microsoft代码示例。由于我过去的尝试失败,我只是复制了网站的代码并更改了端口号(见下文),但我仍然无法接收数据报。

#include <winsock2.h>
#include <ws2tcpip.h>
#include <stdio.h>

int iResult = 0;
WSADATA wsaData;
SOCKET RecvSocket;
sockaddr_in RecvAddr;
unsigned short Port = 18190;

char RecvBuf[1024];
int BufLen = 1024;
sockaddr_in SenderAddr;
int SenderAddrSize = sizeof (SenderAddr);

//-----------------------------------------------
// Initialize Winsock
iResult = WSAStartup(MAKEWORD(2, 2), &wsaData);
if (iResult != NO_ERROR) {
    wprintf(L"WSAStartup failed with error %d\n", iResult);
    return 1;
}
//-----------------------------------------------
// Create a receiver socket to receive datagrams
RecvSocket = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);
if (RecvSocket == INVALID_SOCKET) {
    wprintf(L"socket failed with error %d\n", WSAGetLastError());
    return 1;
}
//-----------------------------------------------
// Bind the socket to any address and the specified port.
RecvAddr.sin_family = AF_INET;
RecvAddr.sin_port = htons(Port);
RecvAddr.sin_addr.s_addr = htonl(INADDR_ANY);

iResult = bind(RecvSocket, (SOCKADDR *) &RecvAddr, sizeof (RecvAddr));
if (iResult != 0) {
    wprintf(L"bind failed with error %d\n", WSAGetLastError());
    return 1;
}
//-----------------------------------------------
// Call the recvfrom function to receive datagrams
// on the bound socket.
wprintf(L"Receiving datagrams...\n");
iResult = recvfrom(RecvSocket,
                   RecvBuf, BufLen, 0, (SOCKADDR *) &SenderAddr, &SenderAddrSize);
if (iResult == SOCKET_ERROR) {
    wprintf(L"recvfrom failed with error %d\n", WSAGetLastError());
}

//-----------------------------------------------
// Close the socket when finished receiving datagrams
wprintf(L"Finished receiving. Closing socket.\n");
iResult = closesocket(RecvSocket);
if (iResult == SOCKET_ERROR) {
    wprintf(L"closesocket failed with error %d\n", WSAGetLastError());
    return 1;
}

//-----------------------------------------------
// Clean up and exit.
wprintf(L"Exiting.\n");
WSACleanup();

使用 Wireshark,我看到消息按预期进入(源 IP 地址为 10.1.1.150) Wireshark 捕获

我已在我的 Windows 10 防火墙中添加了规则,以确保端口 18190 已打开并且不会拒绝在该端口上进出的消息。

许多网站声明,对于客户端,您不应该绑定套接字,但如果我不绑定套接字,recvfrom()则会立即返回 -1 和错误WSAEINVAL,指出“提供了无效参数。”</p>

注意:我想使用recvfrom()而不是recv(),因为我需要发件人地址信息。

谁能帮我理解为什么这段代码没有收到广播的 UDP 数据报?

更新

sendto()我尝试在调用之前通过发送消息recvfrom()。Wireshark 显示消息已发送到 10.1.1.150,但recvfrom()仍然没有收到任何信息(即它只是永远阻塞)。

更新 2

使用上面的代码,如果我将计算机的静态 IP 地址设置为 10.1.1.255,那么我就能够接收数据包。这告诉我代码能够接收数据包,但仍然无法解释为什么我无法接收广播数据包。

标签: c++socketsudprecvfrom

解决方案


所以我终于找到了问题,它很愚蠢。我在这里发布解决方案可能会帮助其他人。

我的子网掩码默认为 255.0.0.0,这对于获取广播数据包是不正确的。将子网掩码更改为 255.255.255.0 允许代码获取广播数据包。


推荐阅读