如何在Java中确定因特网网络接口

2022-09-03 18:04:53

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

InetAddress.getLocalHost().getHostAddress();

在Eclipse中,这完全返回了我的意图,192.168.1.105。但是,如果我将其打包到jar文件中并运行该程序,则代码返回169.254.234.50。对此进行调查后,我发现这是我计算机上 VMware 虚拟以太网适配器接口的 IP 地址。

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

接口比较

接口 [网络4]

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和link local=false,因此这些字段也没有任何帮助。


答案 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的答案用isReachable(...)更新了我的答案。

有一件事要注意。 会对我大喊大叫,如果我尝试连续运行代码太快,就像连接清理得不够快一样,地址和端口已经在使用中。 应该是一个随机端口。socket.bind(...)8080


答案 2

请参阅 Java 教程中的检索网络接口列出网络接口地址

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

更新:

我发现了与你类似的问题。
请参阅如何检查是否在java中存在互联网连接的答案。

更新 2:

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

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

推荐