- 您有多个网络适配器。
- 将 UDP 套接字绑定到本地端口,而不指定地址。
- 在其中一个适配器上接收数据包。
你如何获得接收数据包的适配器的本地 IP 地址?
问题是,“接收适配器的 IP 地址是什么?” 不是我们从发件人那里得到的地址
receive_from( ..., &senderAddr, ... );
称呼。
您可以枚举所有网络适配器,获取它们的 IP 地址,并将子网掩码覆盖的部分与发送者的地址进行比较。
像:
IPAddress FindLocalIPAddressOfIncomingPacket( senderAddr )
{
foreach( adapter in EnumAllNetworkAdapters() )
{
adapterSubnet = adapter.subnetmask & adapter.ipaddress;
senderSubnet = adapter.subnetmask & senderAddr;
if( adapterSubnet == senderSubnet )
{
return adapter.ipaddress;
}
}
}
天,
我假设您已经使用 INADDR_ANY 完成了绑定以指定地址。
如果是这种情况,那么 INADDR_ANY 的语义就是在所有接口上指定的端口上创建一个 UDP 套接字。套接字将获取发送到指定端口上所有接口的所有数据包。
使用此套接字发送时,使用编号最小的接口。传出发送方的地址字段设置为使用的第一个传出接口的 IP 地址。
第一个传出接口定义为执行 ifconfig -a 时的序列。它可能是 eth0。
HTH。
干杯,罗伯
不幸的是,当与绑定到“任何 IP”的套接字一起使用时,sendto 和 recvfrom API 调用从根本上被破坏了,因为它们没有本地 IP 信息的字段。
所以你对此能做些什么?
这些选项都不是很好。猜测显然有时会产生错误的答案。绑定单独的套接字会增加软件的复杂性,并且如果本地地址列表发生更改,您的程序正在运行,则会导致问题。较新的 API 是正确的技术解决方案,但可能会降低可移植性(特别是看起来 WSArecvmsg 在 Windows XP 上不可用),并且可能需要修改您正在使用的套接字包装库。
编辑看起来我错了,似乎 MS 文档具有误导性,并且 WSArecvmsg 在 Windows XP 上可用。见https://stackoverflow.com/a/37334943/5083516
您可以先将 SO_REUSEADDR 设置为 true
BOOL bOptVal = 1;
setsockopt(so, SOL_SOCKET, SO_REUSEADDR, (char *)&boOptVal, sizeof(bOptVal));
创建另一个套接字后receive_from( ..., &remoteAddr, ... );
,并连接回remoteAddr。然后调用getsockname可以得到ip地址。
SOCKET skNew = socket( )
// Same local address and port as that of your first socket
// INADDR_ANY
bind(skNew, , )
// set SO_REUSEADDR to true again
setsockopt(skNew, SOL_SOCKET, SO_REUSEADDR, (char *)&boOptVal, sizeof(bOptVal));
// connect back
connect(skNew, remoteAddr)
// get local address of the socket
getsocketname(skNew, )
大小_t recvfrom(int socket, void *restrict buffer, size_t length, int flags, struct sockaddr *restrict address, socklen_t *restrict address_len); 大小_t recvmsg(int socket, struct msghdr *message, int flags); [..] 如果地址不是空指针并且套接字不是面向连接的,则 填写消息的源地址。
实际代码:
int nbytes = recvfrom(sock, buf, MAXBUFSIZE, MSG_WAITALL, (struct sockaddr *)&bindaddr, &addrlen);
fprintf(stdout, "Read %d bytes on local address %s\n", nbytes, inet_ntoa(bindaddr.sin_addr.s_addr));
希望这可以帮助。
试试这个:
gethostbyname("localhost");