aur*_*amo 6 java linux networking udp broadcast
我有一个奇怪的要求,能够在Linux机器上从Java监听许多网络接口,并确定其中一个是否接收某种类型的UDP数据包.我需要的输出数据是相关接口的IP地址.有没有办法在Java中这样做?
通过通配符地址(新的DatagramSocket(端口))进行监听没有帮助,因为虽然我获得了广播数据包,但我无法确定它们所通过的接口的本地IP地址.在绑定到某个接口(新的DatagramSocket(端口,地址))时收听广播根本不接收数据包.这种情况值得一个代码示例,显示我正在尝试做的事情:
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
}
}
Run Code Online (Sandbox Code Playgroud)
我还尝试使用基于接口的真实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);
Run Code Online (Sandbox Code Playgroud)
这只是在构造DatagramSocket时抛出一个BindException.
编辑: BindException(java.net.BindException:无法分配请求的地址)从调用DatagramSocket的构造函数与广播地址(例如126.255.255.255)只附带最新的Ubuntu 9.04(可能不是Ubuntu,但内核版本特定的问题) .使用Ubuntu 8.10,以及我正在处理的Red Hat版本(RHEL 4.x).
显然在绑定到某个本地IP时没有收到数据包是正确的行为,尽管在Windows中这是有效的.我需要让它在Linux(RHEL和Ubuntu)上运行.对于低级C代码,有一个解决方法setsockopt(SO_BINDTODEVICE),我在Java-API中找不到它.这并不能让我在乐观中突然爆发:-)
这最终是一个 IPV6 Linux 内核问题。通常我会禁用 IPV6,因为它会引起各种头痛。然而在 Ubuntu 9.04 中,禁用 IPV6 太难了,我放弃了,这让我很痛苦。
要监听来自某个接口的广播消息,我将首先创建该接口 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);
Run Code Online (Sandbox Code Playgroud)
当然,如果许多接口都有一个以相同网络部分开头的 IP,这并不会真正将我绑定到某个接口,但对我来说这个解决方案就足够了。
然后我使用该地址(和所需的端口)创建数据报套接字,并且它可以工作。但必须将以下系统属性传递给 JVM:
-Djava.net.preferIPv6Addresses=false -Djava.net.preferIPv4Stack=true
Run Code Online (Sandbox Code Playgroud)
我不知道 IPV6 如何设法破坏广播收听,但确实如此,并且上述参数修复了它。
| 归档时间: |
|
| 查看次数: |
8598 次 |
| 最近记录: |