Подтвердить что ты не робот

Как определить сетевой интерфейс сети в Java

Как вы определяете, какой сетевой интерфейс подключен к Интернету с помощью Java? Например, я запускаю

InetAddress.getLocalHost().getHostAddress();

и в Eclipse это возвращает именно то, что я намереваюсь, 192.168.1.105. Однако, если я упакую это в файл jar и запустил программу, код вернет 169.254.234.50. В этом я обнаружил, что это IP-адрес интерфейса виртуального Ethernet-адаптера VMware на моей машине.

Есть ли способ определить интерфейс, подключенный к Интернету, но в то же время поддерживать переносимость моего кода?

Сравнение интерфейсов

Интерфейс [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 с локальным сайтом = true и link local = false, поэтому эти поля также не помогают.

4b9b3361

Ответ 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 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;
  }
}

(Я обновил свой ответ с помощью isReachable(...) на основе ответа Mocker Tim.)

Одна вещь, на которую нужно следить. socket.bind(...) будет лаять на меня, что адрес и порт уже используются, если я попытался запустить мой код слишком быстро подряд, так как соединение не было очищено достаточно быстро. 8080 должен быть случайным портом.

Ответ 2

Посмотрите Извлечение сетевых интерфейсов и Список сетевого интерфейса Адреса в учебнике Java.

Если у вас появилось несколько сетевых интерфейсов, вы должны программно выбрать сетевой интерфейс, который должна использовать ваша программа.

UPDATE:

Я нашел вопрос, похожий на ваш.
См. Ответ how-to-check-if-internet-connection-is-present-in-java.

ОБНОВЛЕНИЕ 2:

IMHO Вы должны сделать следующее в своей программе:

  • Найдите все сетевые интерфейсы IPv4 на компьютере, на котором запущена ваша программа;
  • Спросите пользователя, какая из них должна использовать ваша программа;
  • Используйте интерфейс, на который указал пользователь.

Ответ 3

У меня была такая же проблема, и я придумал это решение (не так тщательно, как ответ, но мне нужно было что-то, что не пыталось подключиться к Интернету.

На основе известного OUI для адресов mac: 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;
}