InetAddress.getLocalHost() 不可靠,应遍历 NetworkInterface 获取真实网卡 IP;getByName("www.baidu.com") 抛 UnknownHostException 是因 DNS 解析失败;IPv6 地址带 % 后缀需截断或用 InetAddress 对象直连;DNS 缓存需合理配置 TTL。

getByName() 和 getLocalHost() 哪个更适合获取本机IP
直接调用 InetAddress.getLocalHost() 看似最简单,但它常返回 127.0.1.1 或主机名解析失败(尤其在 Docker 容器、/etc/hosts 配置异常、无 DNS 的离线环境),**不可靠**。而 InetAddress.getByName("localhost") 也只保证回环地址,不是真实网卡 IP。
真正需要对外通信的本机 IP,应遍历所有网卡接口:
- 用
NetworkInterface.getNetworkInterfaces()获取全部接口 - 对每个接口调用
getInetAddresses(),过滤掉isAnyLocalAddress()、isLoopbackAddress()、isLinkLocalAddress()的地址 - 优先选
inetAddress instanceof Inet4Address(IPv4 更通用)
为什么 getByName("www.baidu.com") 有时抛 UnknownHostException
这不是代码写错了,而是底层 DNS 解析失败。常见原因包括:
— 当前机器未配置可用 DNS 服务器(如 /etc/resolv.conf 为空或内容无效)
— 目标域名确实无法解析(已过期、拼写错误、DNS 污染)
— JVM 启动时设置了 -Dnetworkaddress.cache.ttl=-1 但 DNS 服务本身不可达,导致首次查询即超时
— 在 Android 或某些受限容器中,getByName() 默认走系统 DNS,但网络策略禁止 UDP 53 端口
解决办法不是重试,而是加超时控制(Java 7+):
try {
InetAddress address = InetAddress.getByName("www.baidu.com");
} catch (UnknownHostException e) {
// 不建议静默吞掉,至少记录日志
}更健壮的做法是用 InetAddress.getAllByName(host) 并设 DNS 缓存策略,或改用 java.net.http.HttpClient + 自定义 DNS 解析器(需额外依赖)。
getHostAddress() 返回 IPv6 地址带 % 符号怎么处理
形如 fe80::1%lo0 的地址是「链路本地 IPv6 地址 + 作用域 ID」,%lo0 表示该地址仅在 lo0(回环接口)有效。这个 % 后缀是 Java 为兼容 RFC 4007 主动添加的,**不能直接用于 socket 连接或 HTTP 请求**——大多数网络库会报 IllegalArgumentException 或连接拒绝。
正确做法是:若确定只需 IP 字符串(比如日志、展示),用 inetAddress.getHostAddress().split("%")[0] 截断;若要发起连接,应使用原始 InetAddress 对象(如 Socket(InetAddress, port)),它内部会自动处理作用域。
立即学习“Java免费学习笔记(深入)”;
缓存行为和 setCachePolicy() 的实际影响
InetAddress 有内置 DNS 缓存,且默认永久缓存成功结果(cache_ttl = -1)、2 秒缓存失败结果(negative_cache_ttl = 2)。这意味着:
— 修改了 hosts 文件或 DNS 服务器,旧程序可能仍用过期 IP 连接数小时
— 频繁调用 getByName() 查不同域名,缓存会持续增长,无自动淘汰机制
— SecurityManager 已废弃,现在只能通过系统属性控制:
java -Dnetworkaddress.cache.ttl=30 -Dnetworkaddress.cache.negative.ttl=5 MyApp
注意:设置 networkaddress.cache.ttl=0 表示禁用缓存,每次都会触发真实 DNS 查询,延迟明显上升;设为 -1 是永久缓存,适合稳定内网环境。生产系统建议显式设为 30(秒)级,平衡一致性与性能。
真实场景里,InetAddress 很少单独用;多数时候它只是 Socket、URL、OkHttp 内部调用的中间层。真正容易出问题的,是假设「getLocalHost() 就是本机出口 IP」,或把带 % 的 IPv6 字符串直接塞进 HTTP Host 头——这些细节不查文档、不看日志,很容易卡一整天。










