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

新手上路,请多包涵

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

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

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

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

  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());
    }
}

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

 127.0.0.1
192.168.1.2
192.168.56.1
117.204.44.19

原文由 sasidhar 发布,翻译遵循 CC BY-SA 4.0 许可协议

阅读 728
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。

编辑:

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

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

原文由 Mr.Wang from Next Door 发布,翻译遵循 CC BY-SA 4.0 许可协议

在大多数情况下,这可能有点棘手。

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

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

你怎么处理这个?

  • 一种方法是使用 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 地址。

原文由 Stephen C 发布,翻译遵循 CC BY-SA 3.0 许可协议

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题