【问题标题】:How to Determine Internet Network Interface in Java如何在 Java 中确定 Internet 网络接口
【发布时间】:2011-12-11 07:10:15
【问题描述】:

如何确定使用 Java 连接到 Internet 的网络接口?例如,我运行

InetAddress.getLocalHost().getHostAddress();

在 Eclipse 中,它返回的正是我想要的,192.168.1.105。但是,如果我将它打包成一个 jar 文件并运行程序,代码返回 169.254.234.50。调查了一下,我发现这是我机器上 VMware 虚拟以太网适配器接口的 IP 地址。

有什么方法可以确定连接到互联网的接口,同时保持我的代码的可移植性?

接口比较

接口 [net4]

display name  : Intel(R) Centrino(R) Ultimate-N 6300 AGN
MTU           : 1500
loopback      : false
point to point: false
up            : true
virtual       : false
multicast     : true
HW address    : 00 24 D7 2C 5F 70 
INET address (IPv4): 192.168.1.105
  host name           : MyComputer
  canonical host name : MyComputer
  loopback            : false
  site local          : true
  any local           : false
  link local          : false
  multicast           : false
  reachable           : true

接口 [eth5]

display name  : VMware Virtual Ethernet Adapter for VMnet1
MTU           : 1500
loopback      : false
point to point: false
up            : true
virtual       : false
multicast     : true
HW address    : 00 50 56 C0 00 01 
INET address (IPv4): 169.254.234.50
  host name           : MyComputer
  canonical host name : MyComputer
  loopback            : false
  site local          : false
  any local           : false
  link local          : true
  multicast           : false
  reachable           : true

还有第三个 VMware 界面,其中站点 local=true 和链接 local=false,因此这些字段也没有任何帮助。

【问题讨论】:

  • 您是否在 VM 中运行 JAR? VM 是它自己的环境,有它自己的网络接口,这是 Java 返回的。
  • 不,这是在 Windows 7 机器上。但是我安装了 VMWare,我相信这是它用来与 VM 通信的接口。我不确定为什么 Java 选择这个接口作为默认接口。这种行为在我室友的电脑上是相同的(Windows 7,安装了 VMWare)
  • @CavynVonDeylen 请查看我的更新答案。我发布的代码只打印出我的真实网络接口,而我已经安装并启用了 Virtual Box 的网络。

标签: java networking network-interface


【解决方案1】:

在我的笔记本电脑(运行 Windows 7,安装了 Virtual Box 和它的网络接口)上,以下代码打印出我的无线接口的名称以及我的本地地址。它在一天结束时使用蛮力方法,但只会尝试并实际连接到被认为是最佳候选者的地址。

// iterate over the network interfaces known to java
Enumeration<NetworkInterface> interfaces = NetworkInterface.getNetworkInterfaces();
OUTER : for (NetworkInterface interface_ : Collections.list(interfaces)) {
  // we shouldn't care about loopback addresses
  if (interface_.isLoopback())
    continue;

  // if you don't expect the interface to be up you can skip this
  // though it would question the usability of the rest of the code
  if (!interface_.isUp())
    continue;

  // iterate over the addresses associated with the interface
  Enumeration<InetAddress> addresses = interface_.getInetAddresses();
  for (InetAddress address : Collections.list(addresses)) {
    // look only for ipv4 addresses
    if (address instanceof Inet6Address)
      continue;

    // use a timeout big enough for your needs
    if (!address.isReachable(3000))
      continue;

    // java 7's try-with-resources statement, so that
    // we close the socket immediately after use
    try (SocketChannel socket = SocketChannel.open()) {
      // again, use a big enough timeout
      socket.socket().setSoTimeout(3000);

      // bind the socket to your local interface
      socket.bind(new InetSocketAddress(address, 8080));

      // try to connect to *somewhere*
      socket.connect(new InetSocketAddress("google.com", 80));
    } catch (IOException ex) {
      ex.printStackTrace();
      continue;
    }

    System.out.format("ni: %s, ia: %s\n", interface_, address);

    // stops at the first *working* solution
    break OUTER;
  }
}

(我已根据Mocker Tim's 答案更新了isReachable(...) 的答案。)

需要注意的一件事。 socket.bind(...) 如果我试图连续太快地运行我的代码,比如连接没有足够快地清理干净,socket.bind(...) 会向我咆哮地址和端口已经在使用中。 8080 应该是一个随机端口。

【讨论】:

  • 这将返回我想要的接口和我计算机上的两个 VMWare 接口。我想要的界面是第一个,但我真的不想一直依赖它。
  • @CavynVonDeylen 如果您能描述您的用例,我们可以提出某种的解决方案。但是,没有 clean 方法可以做到这一点。我的第一个猜测是您应该按 IP 过滤地址:丢弃以192.168.1 开头的地址not。是的,这是野蛮
  • 谢谢!这段代码对我有用。我很惊讶 Java 没有提供一些简单的方法来做到这一点,因为我认为这将是一个相当普遍的问题。也许在 Java 8 中。
  • @CavynVonDeylen 也许在 Java 8 中。 好吧,我对此表示怀疑。尽管我不是网络专家,但据我所知,您无法将网络接口设置为操作系统级别的默认设置。只有应用程序可以使用 some 接口作为默认接口。基本上,如果您要使用上面的代码,将这样做:基于某些假设将使用接口作为您的默认值。无论如何,我还有一件事要注意:也许代码中的 isReachable(...) 部分是多余的,因为此时的接口是 up 并且它们的地址也是本地的,所以你可能可以跳过它.
【解决方案2】:

查看 Java 教程中的 Retrieving Network InterfacesListing Network Interface Addresses

如果您有多个正在运行的网络接口,则必须以编程方式选择程序应使用的网络接口。

更新:

我发现了与您的问题相似的问题。
查看how-to-check-if-internet-connection-is-present-in-java的答案。

更新 2:

恕我直言,您应该在程序中执行以下操作:

  1. 在运行程序的机器上查找所有 IPv4 网络接口;
  2. 询问用户您的程序应该使用哪一个;
  3. 使用用户指向的界面。

【讨论】:

  • 我可以列出我机器上所有接口的详尽列表,但我还没有找到任何方法来以编程方式确定哪个接口连接到 Internet。我假设如果我要硬编码“net4”,它恰好是我机器上的正确接口,这不适用于运行我的程序的其他计算机。
【解决方案3】:

我遇到了同样的问题,并想出了这个解决方案(没有答案那么彻底,但我需要一些不会尝试连接到互联网的东西。

基于 Mac 地址的已知 OUI:http://standards-oui.ieee.org/oui.txt

我做了一个快速检查(可以用二进制逻辑优化)

private static boolean isVmwareMac(byte[] mac) {
    byte invalidMacs[][] = {
            {0x00, 0x05, 0x69},             //VMWare
            {0x00, 0x1C, 0x14},             //VMWare
            {0x00, 0x0C, 0x29},             //VMWare
            {0x00, 0x50, 0x56}              //VMWare
    };

    for (byte[] invalid: invalidMacs){
        if (invalid[0] == mac[0] && invalid[1] == mac[1] && invalid[2] == mac[2])
            return true;
    }

    return false;
}

【讨论】:

    【解决方案4】:

    我被同样的问题困扰了很久,尽管最初的问题是很久以前提出的,但无论如何我还是会附上我对这个问题的看法。 具有讽刺意味的是,答案是与问题一起提供的。
    它是:siteLocal = true
    这对我有用

    1. 获取所有网络接口

      NetworkInterface.getNetworkInterfaces();
      
    2. 挑选出那些当前宕机的(只是减少运行次数)

      if(interface.isUp()){}
      
    3. 获取您刚刚选择的接口的 InetAddresses

      interface.getInetAddresses();
      
    4. 检查这些地址是否是站点本地地址

      if(inetAddress.isSiteLocal()){}
      

    这应该为您提供一个/所有 InetAddress(es) - 以及链接到您的本地网络的网络接口。
    本地站点被定义为使用来自这些网络的地址:10.0.0.0、172.16.0.0、192.168.0.0,而 VirtualBox 适配器和其他通常使用 169.254.0.0 链接本地地址。 IPv6 也实现了站点本地的概念,所以这里最好重点关注 IPv4。 在普通的最终用户机器上,这应该工作得很好,并且只返回一个物理接口。

    【讨论】:

    • 回到这个问题并再次检查了几个论坛并与管理员交谈后,我得出了以下结论。要确定可以访问 Internet 的本地接口,您需要找到与网关在同一网络中的一个(在众多接口中)。显然这无法使用 Java 以编程方式确定,因为无法找到此网关的 IP。检查站点本地接口很接近但不够可靠。所以我猜你无法确定哪个接口使用 Java 连接到互联网。
    • address :- /127.0.0.1 isSiteLocal :- false for 127 我不是本地的。这怎么可能
    • @P Satish Patro 如果您使用 isSiteLocal() 检查它们,三个保留的 IP 地址范围将返回 true:10.0.0.0/8、172.16.0.0/12、192.168.0.0/16 .环回接口是特殊的,因为它用于解决您自己的问题,而不关心已分配给您的主机的地址,也不管分配的地址是这三个私有范围之一的一部分还是可通过互联网路由。简而言之,它不是 3 个 IP 地址范围之一的一部分,因此它将返回 false。
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2021-02-18
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多