工作环境中为啥不能直接使用InetAddress?
一、直接使用getLocalHost()获取本机IP的错误示例
public static void main(String[] args) throws UnknownHostException {
InetAddress localHost = InetAddress.getLocalHost();
System.out.println(localHost.getHostAddress());
}
运行程序,在不同的操作系统上值可能还不一样:
在windows上:192.168.199.175
在windows上看似正常。但是,请你开一个vpn再运行试一下,定会让你大跌眼镜(比如我开启公司的vpn后,输出的值是1.0.0.137)
可以看到当出现多个网卡接口工作时,windows可能就不好使了,而多个网卡同时工作的情况是很正常的(比较虚拟网卡经常很多)在Linux上:127.0.0.1,what?
为何在Linux下请你一定不要使用它来获取本机IP,因为它就是简单的读取/etc/hosts的内容,所以它默认返回的是127.0.0.1非常的不靠谱,因此本方法十分不建议在生产上使用。
二、获取本机IP地址的正确姿势
简单情况(废弃)
简单情况下,就可以通过InetAddress.getLocalHost()来获取到本机ip地址。注意这里的关键词:简单。因此它对环境是有要求的:
windows环境
非多网卡协同工作环境(比如不能开启vpn)
很明显,这种“简单”情况在实际生产中并不存在,因此仅限yy,不可使用。
复杂情况(通用,推荐的方案)
public static void main(String[] args) throws UnknownHostException {
InetAddress localHost = InetAddress.getLocalHost();
System.out.println(localHost.getHostAddress());
System.out.println("----------------下面才是正确的获取方式----------------");
localHost = getLocalHostExactAddress();
System.out.println(localHost.getHostAddress());
// System.out.println(localHost.getHostName());
}
public static InetAddress getLocalHostExactAddress() {
try {
InetAddress candidateAddress = null;
Enumeration<NetworkInterface> networkInterfaces = NetworkInterface.getNetworkInterfaces();
while (networkInterfaces.hasMoreElements()) {
NetworkInterface iface = networkInterfaces.nextElement();
// 该网卡接口下的ip会有多个,也需要一个个的遍历,找到自己所需要的
for (Enumeration<InetAddress> inetAddrs = iface.getInetAddresses(); inetAddrs.hasMoreElements(); ) {
InetAddress inetAddr = inetAddrs.nextElement();
// 排除loopback回环类型地址(不管是IPv4还是IPv6 只要是回环地址都会返回true)
if (!inetAddr.isLoopbackAddress()) {
if (inetAddr.isSiteLocalAddress()) {
// 如果是site-local地址,就是它了 就是我们要找的
// ~~~~~~~~~~~~~绝大部分情况下都会在此处返回你的ip地址值~~~~~~~~~~~~~
return inetAddr;
}
// 若不是site-local地址 那就记录下该地址当作候选
if (candidateAddress == null) {
candidateAddress = inetAddr;
}
}
}
}
// 如果出去loopback回环地之外无其它地址了,那就回退到原始方案吧
return candidateAddress == null ? InetAddress.getLocalHost() : candidateAddress;
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
在windows运行输出(即使你开了vpn都没有关系):
2.0.2.81
----------------下面才是正确的获取方式----------------
192.168.199.175
在Linux下运行输出:
127.0.0.1
----------------下面才是正确的获取方式----------------
10.102.11.29
可以看到通过这种方式获取本机IP,不管是windows or Linux,亦不管是开启了vpn与否,最终均能得到正确的结果。
下面还有一些其他方法(InetAddress)
//获取主机名
public String getHostName();
//获取主机名(每次方法都解析DNS去获取主机名,而不从缓存中获取)
public String getCanonicalHostName();
//获取以字节数组形式的ip地址。
public byte[] getAddress();
//获取点分四段式ip。
public String getHostAddress();
//可以根据指定timeout时间内测试网络是否可达,是否可以建立连接。
public boolean isReachable(int timeout);
//可以指定从本机那个网卡、生存时间ttl 和指定时间来测试网络是否可达。
public boolean isReachable(NetworkInterface netif, int ttl, int timeout);
// 当IP地址是广播地址(MulticastAddress)时返回true,否则返回false。
// 通过广播地址可以向网络中的所有计算机发送信息,而不是只向一台特定的计算机发送信息。
// IPv4的广播地址的范围是224.0.0.0 ~ 239.255.255.255
// IPv6的广播地址第一个字节是FF 其他的字节可以是任意值。
public boolean isMulticastAddress();
// 当IP地址是通配符地址时返回true,否则返回false。
// IPv4的通配符地址是0.0.0.0
// IPv6的通配符地址是0:0:0:0:0:0:0:0,也可以简写成::。
public boolean isAnyLocalAddress();
// 当IP地址是loopback地址时返回true,否则返回false。loopback地址就是代表本机的IP地址。
// IPv4的loopback地址的范围是127.0.0.0 ~ 127.255.255.255,也就是说,只要第一个字节是127,就是lookback地址。
// IPv6的loopback地址是0:0:0:0:0:0:0:1,也可以简写成::1
public boolean isLoopbackAddress();
// 当IP地址是本地连接地址(LinkLocalAddress)时返回true,否则返回false。
// IPv4的本地连接地址的范围是169.254.0.0 ~ 169.254.255.255。
// IPv6的本地连接地址的前12位是FE8,其他的位可以是任意取值,如FE88::和FE80::ABCD::都是本地连接地址。
public boolean isLinkLocalAddress();
// 当IP地址是地区本地地址(SiteLocalAddress)时返回true,否则返回false。(是不是内网ip)
// IPv4的地址本地地址分为三段:10.0.0.0 ~ 10.255.255.255、172.16.0.0 ~ 172.31.255.255、192.168.0.0 ~ 192.168.255.255
// (企业内部或个人内部的局域网内部的ip都应该在此三个网段内 因为内网用ipv4足够)
// IPv6的地区本地地址的前12位是FEC,其他的位可以是任意取值,如FED0:: 和 FEF1:: 都是地区本地地址。
public boolean isSiteLocalAddress();
// 当IP地址是全球范围的广播地址时返回true,否则返回false。全球范围的广播地址可以向Internet中的所有的计算机发送信息。
// IPv4的广播地址除了224.0.0.0和第一个字节是239的IP地址都是全球范围的广播地址
// IPv6的全球范围的广播地址中第一个字节是FF,第二个字节的范围是0E ~ FE,其他的字节可以是任意值,如FFBE::、FF0E::都是全球范围的广播地址。
public boolean isMCGlobal();
// 当IP地址是本地接口广播地址时返回true,否则返回false。
// 本地接口广播地址不能将广播信息发送到产生广播信息的网络接口,即使是同一台计算机的另一个网络接口也不行。
// 所有的IPv4广播地址都不是本地接口广播地址。IPv6的本地接口广播地址的第一个字节是FF,第二个节字的范围是01 ~ F1,
// 其他的字节可以是任意值,如FFB1::、FF01:A123::都是本地接口广播地址。
public boolean isMCNodeLocal();
// 当IP地址是子网广播地址时返回true,否则返回false。
// IPv4的子网广播地址的范围是224.0.0.0 ~ 224.0.0.255。
// IPv6的子网广播地址的第一个字节是FF,第二个字节的范围是02 ~ F2,其他的字节可以是任意值,如FFB2:: 和 FF02:ABCD:: 都是子网广播地址。
public boolean isMCLinkLocal();
// 当IP地址是站点范围的广播地址时返回true,否则返回false。
// 使用站点范围的广播地址,可以向站点范围内的计算机发送广播信息。
// IPv4的站点范围广播地址的范围是239.255.0.0 ~ 239.255.255.255,如239.255.1.1、239.255.0.0都是站点范围的广播地址。
// IPv6的站点范围广播地址的第一个字节是FF,第二个字节的范围是05 ~ F5,其他的字节可以是任意值,如FF05:: 和 FF45:: 都是站点范围的广播地址。
public boolean isMCSiteLocal();
// 当IP地址是组织范围的广播地址时返回ture,否则返回false。
// 使用组织范围广播地址可以向公司或企业内部的所有的计算机发送广播信息。
// IPv4的组织范围广播地址的第一个字节是239,第二个字节不小于192,第三个字节不大于195,如239.193.100.200、239.192.195.0都是组织范围广播地址。
// IPv6的组织范围广播地址的第一个字节是FF,第二个字节的范围是08 ~ F8,其他的字节可以是任意值,如FF08:: 和 FF48::都是组织范围的广播地址。
public boolean isMCOrgLocal();