我有一个有点奇怪的要求,即能够在 Linux 机器上监听来自 Java 的多个网络接口,并确定其中一个是否接收到某种类型的 UDP 数据包。我需要的输出数据是相关接口的 IP 地址。有没有办法在Java中做到这一点?
监听通配符地址 (new DatagramSocket(port)) 并没有帮助,因为虽然我确实收到了广播数据包,但我无法确定它们通过的接口的本地 IP 地址。在绑定到某个接口(new DatagramSocket(port, address))时收听广播根本不会收到数据包。这个案例值得一个代码示例来展示我正在尝试做的事情:
Enumeration interfaces = NetworkInterface.getNetworkInterfaces();
while (interfaces.hasMoreElements()) {
NetworkInterface ni = (NetworkInterface) interfaces.nextElement();
Enumeration addresses = ni.getInetAddresses();
while (addresses.hasMoreElements()) {
InetAddress address = (InetAddress)addresses.nextElement();
if (address.isLoopbackAddress() || address instanceof Inet6Address)
continue; //Not interested in loopback or ipv6 this time, thanks
DatagramSocket socket = new DatagramSocket(PORT, address);
//Try to read the broadcast messages from socket here
}
}
我还尝试使用基于接口的真实 IP 开头构造的广播地址初始化套接字,其余的根据正确的网络掩码进行初始化:
byte [] mask = { (byte)255, 0, 0, 0 };
byte[] addrBytes = InetAddress.getByName("126.5.6.7").getAddress();
for (int i=0; i < 4; i++) {
addrBytes[i] |= ((byte)0xFF) ^ mask[i];
}
InetAddress bcastAddr = InetAddress.getByAddress(addrBytes);
这只是在构造 DatagramSocket 时抛出一个 BindException。
编辑:使用广播地址(例如 126.255.255.255)调用 DatagramSocket 的构造函数的 BindException(java.net.BindException:无法分配请求的地址)仅随最新的 Ubuntu 9.04 提供(可能不是 Ubuntu,但内核版本特定的问题) . 使用 Ubuntu 8.10,以及我正在处理的 Red Hat 版本(RHEL 4.x)。
显然,在绑定到某个本地 IP 时不接收数据包是正确的行为,尽管在 Windows 中这是有效的。我需要让它在 Linux(RHEL 和 Ubuntu)上运行。对于低级 C 代码,有一种解决方法 setsockopt(SO_BINDTODEVICE),我在 Java-API 中找不到它。不过,这并不完全让我充满乐观:-)