Как определить сетевой интерфейс сети в 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, поэтому эти поля также не помогают.
Ответы
Ответ 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;
}