加入收藏 | 设为首页 | 会员中心 | 我要投稿 衡阳站长网 (https://www.0734zz.cn/)- 数据集成、设备管理、备份、数据加密、智能搜索!
当前位置: 首页 > 服务器 > 搭建环境 > Linux > 正文

Linux上的Java:在绑定的本地地址上侦听广播消息

发布时间:2021-01-15 01:14:28 所属栏目:Linux 来源:网络整理
导读:我有一个奇怪的要求,能够在Linux机器上从Java监听许多网络接口,并确定其中一个是否接收某种类型的UDP数据包.我需要的输出数据是相关接口的IP地址.有没有办法在Java中这样做? 通过通配符地址(新的DatagramSocket(端口))进行监听没有帮助,因为虽然我获得了广

我有一个奇怪的要求,能够在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
  }
}

我还尝试使用基于接口的真实IP的开头构建的广播地址来初始化套接字,其余的根据正确的网络掩码:

byte [] mask = { (byte)255,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.

编辑:BindException(java.net.BindException:无法分配请求的地址)从调用DatagramSocket的构造函数与广播地址(例如126.255.255.255)只附带最新的Ubuntu 9.04(可能不是Ubuntu,但内核版本特定的问题) .使用Ubuntu 8.10,以及我正在处理的Red Hat版本(RHEL 4.x).

显然没有在绑定到某个本地IP时收到数据包是correct behaviour,虽然在Windows中这是有效的.我需要让它在Linux(RHEL和Ubuntu)上运行.对于低级C代码,有一个解决方法setsockopt(SO_BINDTODEVICE),我在Java-API中找不到它.尽管如此,This并没有让我充满乐观:-) 最佳答案 这最终是一个IPV6 Linux内核问题.通常我已禁用IPV6,因为它会导致各种令人头疼的问题.但是在Ubuntu 9.04中我很难禁用我放弃的IPV6,这就是我的意思.

要收听来自某个界面的广播消息,我将首先创建该接口IP地址的“广播版本”:

byte [] mask = { (byte)255,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);

当然,如果许多接口具有以相同网络部分开头的IP,这并不能真正将我绑定到某个接口,但对我来说这个解决方案就足够了.

然后我用该地址(和所需的端口)创建datagramsocket,它的工作原理.但不是没有将以下系统属性传递给JVM:

-Djava.net.preferIPv6Addresses=false -Djava.net.preferIPv4Stack=true 

我不知道IPV6如何设法打破广播的监听,但确实如此,以上参数修复了它.

(编辑:衡阳站长网)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    热点阅读