Как определить интернет-интерфейс сети в Java - PullRequest
12 голосов
/ 11 декабря 2011

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

Ответы [ 4 ]

20 голосов
/ 11 декабря 2011

На моем ноутбуке (под управлением 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;
  }
}

(я обновил свой ответс isReachable(...) на основе Ответа издателя Тима .

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

5 голосов
/ 11 декабря 2011

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

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

UPDATE:

Я нашел вопрос, похожий на ваш.
См. Ответ на вопрос как проверить наличие соединения с интернетом в java .

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

ИМХО В вашей программе вы должны сделать следующее:

  1. Найдите все сетевые интерфейсы IPv4 на компьютере, на котором работает ваша программа;
  2. Спросите пользователя, какую из них должна использовать ваша программа;
  3. Используйте интерфейс, на который указал пользователь.
1 голос
/ 04 декабря 2015

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

На основе известного 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;
}
0 голосов
/ 11 мая 2018

Меня довольно долго жаловала та же проблема, и хотя хотя бы давным-давно был задан начальный вопрос, я все равно добавлю свое мнение по этому вопросу. Ирония в том, что ответ был дан вместе с вопросом.
Это: siteLocal = true
Вот что у меня сработало

  1. Получить все сетевые интерфейсы

    NetworkInterface.getNetworkInterfaces();
    
  2. Выделите те, которые в данный момент отключены (просто уменьшает количество прогонов)

    if(interface.isUp()){}
    
  3. Получите InetAddresses только что выделенных интерфейсов

    interface.getInetAddresses();
    
  4. Проверьте эти адреса, если они являются локальными адресами сайта

    if(inetAddress.isSiteLocal()){}
    

Это должно дать вам один / все InetAddress (и) - и сетевой интерфейс (ы), которые связаны с вашей локальной сетью.
Локальный сайт определен для использования адресов из этих сетей: 10.0.0.0, 172.16.0.0, 192.168.0.0, в то время как адаптеры VirtualBox и другие обычно используют локальные 169.254.0.0 адреса ссылок. IPv6 также реализует локальную концепцию сайта, поэтому лучше сосредоточиться на IPv4 здесь. На обычном компьютере конечного пользователя это должно работать довольно хорошо и возвращать только один физический интерфейс.

...