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

2022-08-31 04:58:28

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

现在,所有节点都会创建一个套接字,其目标 IP 作为称为引导节点的特殊节点的 IP。然后,节点创建自己的节点并开始侦听连接。ServerSocket

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

现在我需要的是节点必须将其IP注册到引导节点。我尝试使用客户端连接到引导节点,但这不起作用。cli.getInetAddress()ServerSocket

  1. 我需要客户注册其PPP IP(如果可用);
  2. 否则,局域网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());
    }
}

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

127.0.0.1
192.168.1.2
192.168.56.1
117.204.44.19

答案 1

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

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

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

你怎么能处理这个问题?

  • 一种方法是用于获取主机上所有已知的网络接口,然后迭代每个NI的地址。NetworkInterface.getNetworkInterfaces()
  • 另一种方法是(以某种方式)获取主机的外部广告化 FQDN,并用于查找主 IP 地址。(但是您如何获得它,以及如何处理基于DNS的负载均衡器?InetAddress.getByName()
  • 上述方法的变体是从配置文件或命令行参数获取首选 FQDN。
  • 另一种变体是从配置文件或命令行参数中获取首选 IP 地址。

总之,通常可以正常工作,但您可能需要为代码在具有“复杂”网络的环境中运行的情况提供另一种方法。InetAddress.getLocalHost()


我能够获得与所有网络接口关联的所有 IP 地址,但我如何区分它们?

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

实际上,InetAddress API 提供了用于测试环回、链接本地、站点本地、多播和广播地址的方法。您可以使用这些来整理您返回的哪些IP地址最合适。


答案 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套接字上具有以下效果:它设置Send/Recv的目标,丢弃来自其他地址的所有数据包,并且 - 这就是我们使用的 - 将套接字传输到“已连接”状态,设置其适当的字段。这包括根据系统的路由表检查到目标的路由是否存在,并相应地设置本地终结点。最后一部分似乎没有正式记录,但它看起来像是伯克利套接字API的一个整体特征(UDP“连接”状态的副作用),它在Windows和Linux中跨版本和发行版可靠地工作。

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

编辑:

正如@macomgil所说,对于MacOS,您可以这样做:

Socket socket = new Socket();
socket.connect(new InetSocketAddress("google.com", 80));
System.out.println(socket.getLocalAddress());

推荐