一尘不染

使用Java获取当前计算机的IP地址

java

我正在尝试开发一个系统,其中有不同的节点在不同的系统或同一系统的不同端口上运行。

现在,所有节点都使用目标IP创建一个套接字,该目标IP是称为自举节点的特殊节点的IP。然后,节点创建自己的节点ServerSocket并开始侦听连接。

引导节点维护节点列表,并在查询时返回它们。

现在,我需要的是该节点必须将其IP注册到引导节点。cli.getInetAddress()一旦客户端连接到ServerSocketBootstrapping节点,我就尝试使用该方法,但这没有用。

  1. 我需要客户端注册其PPP IP(如果有);
  2. 否则,LAN IP(如果有);
  3. 否则,它必须使用相同的计算机注册127.0.0.1。

使用代码:

System.out.println(Inet4Address.getLocalHost().getHostAddress());

要么

System.out.println(InetAddress.getLocalHost().getHostAddress());

我的PPP连接IP地址是:117.204.44.192但以上返回我192.168.1.2

编辑

我正在使用以下代码:

Enumeration e = NetworkInterface.getNetworkInterfaces();
while(e.hasMoreElements())
{
    NetworkInterface n = (NetworkInterface) e.nextElement();
    Enumeration ee = n.getInetAddresses();
    while (ee.hasMoreElements())
    {
        InetAddress i = (InetAddress) ee.nextElement();
        System.out.println(i.getHostAddress());
    }
}

我可以获取所有与all关联的IP地址NetworkInterface,但是如何区分它们呢?这是我得到的输出:

127.0.0.1
192.168.1.2
192.168.56.1
117.204.44.19

阅读 1383

收藏
2020-02-27

共2个答案

一尘不染

import java.net.DatagramSocket;
import java.net.InetAddress;

try(final DatagramSocket socket = new DatagramSocket()){
  socket.connect(InetAddress.getByName("8.8.8.8"), 10002);
  ip = socket.getLocalAddress().getHostAddress();
}

当存在多个网络接口时,这种方法效果很好。它总是返回首选的出站IP。目的地8.8.8.8不需要是可到达的。

Connect在UDP套接字上具有以下效果:它设置发送/接收的目的地,丢弃来自其他地址的所有数据包,并且-我们使用的是-将套接字转换为“已连接”状态,并设置其适当的字段。这包括根据系统的路由表检查到目的地的路由的存在,并相应地设置本地端点。最后一部分似乎没有正式记录,但看起来像Berkeley套接字API的一个固有特征(UDP“已连接”状态的副作用),可以在Windows和Linux的各个版本和发行版中可靠地工作。

因此,此方法将提供用于连接到指定远程主机的本地地址。没有建立真正的连接,因此指定的远程IP可能无法访问。

编辑:

对于MacOS,你可以执行以下操作:

Socket socket = new Socket();
socket.connect(new InetSocketAddress("google.com", 80));
System.out.println(socket.getLocalAddress());
2020-02-27
一尘不染

在最一般的情况下,这可能会有些棘手。

表面上,InetAddress.getLocalHost()应该给你这个主机的IP地址。问题在于主机可能具有许多网络接口,并且一个接口可能绑定到多个IP地址。最重要的是,并非所有IP地址都可以在你的计算机或LAN外部访问。例如,它们可以是虚拟网络设备的IP地址,专用网络IP地址等等。

这意味着返回的IP地址InetAddress.getLocalHost()可能不是正确的使用地址。

你怎么处理呢?

  • 一种方法是用于NetworkInterface.getNetworkInterfaces()获取主机上所有已知的网络接口,然后遍历每个NI的地址。
  • 另一种方法是(以某种方式)获取主机的外部广告FQDN,并用于InetAddress.getByName()查找主IP地址。(但是,如何获得它,以及如何处理基于DNS的负载平衡器?)
  • 先前版本的一种变化是从配置文件或命令行参数中获取首选的FQDN。
  • 另一个变化是从配置文件或命令行参数获取首选的IP地址。
    总之,InetAddress.getLocalHost()通常可以使用,但是对于在具有“复杂”网络的环境中运行代码的情况,你可能需要提供一种替代方法。

我可以获得与所有网络接口相关的所有IP地址,但是如何区分它们呢?

  • 127.xxx.xxx.xxx范围内的任何地址都是“回送”地址。仅对“此”主机可见。
  • 192.168.xxx.xxx范围内的任何地址都是私有(也称为本地站点)IP地址。这些保留供组织内部使用。这同样适用于10.xxx.xxx.xxx地址和172.16.xxx.xxx至172.31.xxx.xxx。
  • 范围为169.254.xxx.xxx的地址是链接本地IP地址。这些保留供单个网段使用。
  • 范围为224.xxx.xxx.xxx到239.xxx.xxx.xxx的地址是多播地址。
  • 地址255.255.255.255是广播地址。
  • 其他任何内容都应该是有效的公共点对点IPv4地址。

实际上,InetAddress API提供了用于测试回送,链接本地,站点本地,多播和广播地址的方法。你可以使用这些来找出最合适的IP地址。

2020-02-27