获取本机的IP地址,看似简单的获取,实则蕴含非常多的操作
这篇文章讲述了PowerJob获取本地IP离奇曲折的经过,以及开放了诸多的可配置参数,打开了我新世界的大窗户。求个关注,求个点赞,求一个评论。
获取地址的操作,本来不应该作为什么重点,但是因为一点小小的意外,导致我对这个环节格外的研究了一下,所以就总结了一下。
先来一段文字,描述一下大致的流程,然后再从源代码中研究一下:
-
先是判断内存是否已经存了IP地址,如果存了,则直接返回保存了个IP地址。
-
从jvm虚拟机中获取配置绑定的IP地址,如果绑定了,则直接返回绑定的IP地址。
-
获取所有的网卡信息,进行遍历。
-
忽略一些无效的网卡信息,比如:虚拟机网口,关闭的网口,启动时配置的忽略网口(主要是用过网卡名字和描述名字)。
-
通过启动时的配置“powerjob.network.interface.preferred”,获取对应的地址,如果有,则直接返回对应的IP地址。
-
对剩余的网卡信息进行遍历,如果遍历到的IP地址合法有效(格式正确并可以ping通),则直接返回该合法有效的ip。
-
直接返回第一条格式正确的IP地址。
-
获取InetAddress.getLocalHost()得到的IP地址。
经过上述一系列的复杂操作,如果没有配置的话,获得到IP地址可能会无效。
简单的开端
public static String getLocalHost() {
//1.先是判断内存是否已经存了IP地址,如果存了,则直接返回保存了个IP地址。if (HOST_ADDRESS != null) {return HOST_ADDRESS;}//2.从jvm虚拟机中获取配置绑定的IP地址,如果绑定了,则直接返回绑定的IP地址。String addressFromJVM = System.getProperty(PowerJobDKey.BIND_LOCAL_ADDRESS);if (StringUtils.isNotEmpty(addressFromJVM)) {log.info("[Net] use address from[{}]: {}", PowerJobDKey.BIND_LOCAL_ADDRESS, addressFromJVM);return HOST_ADDRESS = addressFromJVM;}//第三步在这个方法里面,但是还需要不断的深入才能找到!InetAddress address = getLocalAddress();if (address != null) {return HOST_ADDRESS = address.getHostAddress();}return LOCALHOST_VALUE;
}
一切的开端都是从上面的代码开始的。开始很简单,过程却很复杂。
曲折的经过
第1,2步已经出现,这第3步的出现却需要层层的传送~
public static InetAddress getLocalAddress() {//这个方法只是一个传送门,将其传送到getLocalAddress0
}private static InetAddress getLocalAddress0() {//这个方法也只是传送门,不过这个方法在之后还会出现的InetAddress addressOp = getFirstReachableInetAddress( findNetworkInterface());... ...return localAddress;
}public static NetworkInterface findNetworkInterface() {//传送门依旧,该方法之后也会再次出现List<NetworkInterface> validNetworkInterfaces = emptyList();try {validNetworkInterfaces = getValidNetworkInterfaces();} catch (Throwable e) {log.warn("[Net] findNetworkInterface failed", e);}... ...
}private static List<NetworkInterface> getValidNetworkInterfaces() throws SocketException {List<NetworkInterface> validNetworkInterfaces = new LinkedList<>();//3.获取所有的网卡信息,进行遍历。Enumeration<NetworkInterface> interfaces = NetworkInterface.getNetworkInterfaces();//4.忽略一些无效的网卡信息,比如:虚拟机网口,关闭的网口,//启动时配置的忽略网口(主要是用过网卡名字和描述名字)。while (interfaces.hasMoreElements()) {NetworkInterface networkInterface = interfaces.nextElement();if (ignoreNetworkInterface(networkInterface)) {continue;}// 根据用户 -D 参数忽略网卡if (ignoreInterfaceByConfig(networkInterface.getDisplayName()) || ignoreInterfaceByConfig(networkInterface.getName())) {continue;}validNetworkInterfaces.add(networkInterface);}return validNetworkInterfaces;
}//忽略的网卡内容
private static boolean ignoreNetworkInterface(NetworkInterface networkInterface) throws SocketException {return networkInterface == null|| networkInterface.isLoopback()|| networkInterface.isVirtual()|| !networkInterface.isUp();
}
static boolean ignoreInterfaceByConfig(String interfaceName) {String regex = System.getProperty(PowerJobDKey.IGNORED_NETWORK_INTERFACE_REGEX);if (StringUtils.isBlank(regex)) {return false;}if (interfaceName.matches(regex)) {log.info("[Net] ignore network interface: {} by regex({})", interfaceName, regex);return true;}return false;
}
精彩的高潮
找到了本地所有的网卡信息,并且忽略掉了很多没有用的网卡信息,接下来就是通过偏好来选择合适的网卡地址来进行通信,一开始我没有发现这一条信息,在官方文档中也没有找到对应的配置,一度以为这个ip地址无法选择,甚至我使用的服务器,第一条网卡信息是docker的,结果就直接给我用的docker的地址,直接给我整混乱了,还好我还能看懂这么一点代码,不得不说,作者这个代码确实厉害,直接打开我新世界的大门,但是你开门开的好,你得跟我说一声啊,你不说我都没法往门里进啊。
public static NetworkInterface findNetworkInterface() {List<NetworkInterface> validNetworkInterfaces = emptyList();try {validNetworkInterfaces = getValidNetworkInterfaces();} catch (Throwable e) {log.warn("[Net] findNetworkInterface failed", e);}// Try to find the preferred onefor (NetworkInterface networkInterface : validNetworkInterfaces) {if (isPreferredNetworkInterface(networkInterface)) {log.info("[Net] use preferred network interface: {}", networkInterface.getDisplayName());return networkInterface;}}...return first(validNetworkInterfaces);
}public static boolean isPreferredNetworkInterface(NetworkInterface networkInterface) {
//5.通过启动时的配置“powerjob.network.interface.preferred”,获取对应的地址,如果有,则直接返回对应的IP地址。String preferredNetworkInterface = System.getProperty(PowerJobDKey.PREFERRED_NETWORK_INTERFACE);if (Objects.equals(networkInterface.getDisplayName(), preferredNetworkInterface)) {return true;}// 兼容直接使用网卡名称的情况,比如 Realtek PCIe GBE Family Controllerreturn Objects.equals(networkInterface.getName(), preferredNetworkInterface);
}
无奈的结局
我认为通过偏好选择网卡信息就已经非常好了,如果没有偏好设置,默认选择第一条网卡信息这个策略也是不错,之后是选择一条能够访问的地址,最后如果都不行,就破罐子破摔的来获取一个地址,反正必须要返回一个地址,即使这个地址有问题,也得返回了,如果要是一般的网络环境,我觉得这也挺好的,万一有那么一个公司,内网互相通讯还需要代理,这可就恶心了,太恶心了。
public static NetworkInterface findNetworkInterface() {List<NetworkInterface> validNetworkInterfaces = emptyList();try {validNetworkInterfaces = getValidNetworkInterfaces();} catch (Throwable e) {log.warn("[Net] findNetworkInterface failed", e);}// Try to find the preferred onefor (NetworkInterface networkInterface : validNetworkInterfaces) {if (isPreferredNetworkInterface(networkInterface)) {log.info("[Net] use preferred network interface: {}", networkInterface.getDisplayName());return networkInterface;}}
//6.对剩余的网卡信息进行遍历,如果遍历到的IP地址合法有效(格式正确并可以ping通),则直接返回该合法有效的ip。for (NetworkInterface networkInterface : validNetworkInterfaces) {InetAddress addressOp = getFirstReachableInetAddress(networkInterface);if (addressOp != null) {return networkInterface;}}
//7.直接返回第一条格式正确的IP地址。return first(validNetworkInterfaces);
}private static InetAddress getFirstReachableInetAddress(NetworkInterface networkInterface) {if(networkInterface == null ){return null;}Enumeration<InetAddress> addresses = networkInterface.getInetAddresses();while (addresses.hasMoreElements()) {Optional<InetAddress> addressOp = toValidAddress(addresses.nextElement());if (addressOp.isPresent()) {try {if (addressOp.get().isReachable(100)) {return addressOp.get();}} catch (IOException e) {// ignore}}}return null;
}public static <T> T first(Collection<T> values) {if (values == null || values.isEmpty()) {return null;}if (values instanceof List) {List<T> list = (List<T>) values;return list.get(0);} else {return values.iterator().next();}
}
private static InetAddress getLocalAddress0() {// @since 2.7.6, choose the {@link NetworkInterface} firsttry {InetAddress addressOp = getFirstReachableInetAddress( findNetworkInterface());if (addressOp != null) {return addressOp;}} catch (Throwable e) {log.warn("[Net] getLocalAddress0 failed.", e);}InetAddress localAddress = null;try {
//8.获取InetAddress.getLocalHost()得到的IP地址。localAddress = InetAddress.getLocalHost();Optional<InetAddress> addressOp = toValidAddress(localAddress);if (addressOp.isPresent()) {return addressOp.get();}} catch (Throwable e) {log.warn("[Net] getLocalAddress0 failed.", e);}return localAddress;
}
总结
其实大部分人是不需要了解这部分代码的,基本都不会有啥问题,因为大部分人使用的都是正常人使用的网络,只有我们公司这1万来人用的是不正常人使用的网络,但是万一遇到这方面的问题,了解一下还是好的。
相关文章:

获取本机的IP地址,看似简单的获取,实则蕴含非常多的操作
这篇文章讲述了PowerJob获取本地IP离奇曲折的经过,以及开放了诸多的可配置参数,打开了我新世界的大窗户。求个关注,求个点赞,求一个评论。 获取地址的操作,本来不应该作为什么重点,但是因为一点小小的意外&…...

【SSM】篇一:初试Spring--Ioc与Bean
文章目录1、Spring2、SpringFramework系统架构3、BeanBean的配置Bean的实例化Bean的生命周期4、依赖注入DIsetter注入和构造器注入依赖自动装配5、集合注入1、Spring Spring地址:https://spring.io Spring技术的优点: Spring家族(Spring全家…...

华为OD机试真题Python实现【出租车计费】真题+解题思路+代码(20222023)
🔥系列专栏 华为OD机试(Python)真题目录汇总华为OD机试(JAVA)真题目录汇总华为OD机试(C++)真题目录汇总华为OD机试(JavaScript)真题目录汇总文章目录 🔥系列专栏题目输入输出示例一输入输出说明示例二输入输出说明...

Elasticsearch:如何修改 nested 字段的值
Nested 类型是 object 数据类型的特殊版本,它允许对象数组以一种可以彼此独立查询的方式进行索引。在内部,嵌套对象将数组中的每个对象索引为单独的隐藏文档,这意味着每个嵌套对象都可以使用 nested query 独立于其他对象进行查询。每个 nest…...

【JAVA】jdk8 Stream 排序精通
背景 jdk8的stream流能方便的排序,但是每次都要查资料,非常不方便,不确定,所以这次直接弄懂,不再迷茫。 转载请注明来源,创作不易,请多多支持。 基础排序 stream流 大家应该都比较熟悉了&…...

python的opencv操作记录12——Canny算子使用
文章目录Canny算子非极大值抑制非极大值抑制中的插值滞后阈值实际应用直接使用Canny算子使用膨胀先阈值分割Canny算子 上一篇说到,我在一个小项目里需要在一幅图像中提取一根试管里的两种液体的截面。为了达到这个目的使用传统图像里的区域分割技术,实际…...

Spark on hive Hive on spark
文章目录Spark on hive & Hive on sparkHive 架构与基本原理Spark on hiveHive on sparkSpark on hive & Hive on spark Hive 架构与基本原理 Hive 的核心部件主要是 User Interface(1)和 Driver(3)。而不论是元数据库&a…...

【MySQL】子查询
这里写自定义目录标题子查询1、子查询的基本使用2、 单行子查询2.1、单行比较查询2.2、HAVING 中的子查询2.3、CASE中的子查询3、多行子查询4、相关子查询5、EXISTS 与 NOT EXISTS关键字子查询 子查询指一个查询语句嵌套在另一个查询语句内部的查询,这个特性从MySQ…...

Day889.MySQL高可用 -MySQL实战
MySQL高可用 Hi,我是阿昌,今天学习记录的是关于MySQL高可用的内容。 正常情况下,只要主库执行更新生成的所有 binlog,都可以传到备库并被正确地执行,备库就能达到跟主库一致的状态,这就是最终一致性。但是…...

剑指 Offer 24. 反转链表
⭐简单说两句⭐ CSDN个人主页:后端小知识 🔎GZH:后端小知识 🎉欢迎关注🔎点赞👍收藏⭐️留言📝 题目: 剑指 Offer 24. 反转链表 ,我们今天还是来看一道easy的题目吧&…...

“黑铁时代”,地产人如何以客户视角加速房企数字化转型
本文从行业洞察、业务设计、数据建设以及实践探索四个部分详细阐述地产行业数字化的实践、思考和理解。点击文末“阅读原文”,观看完整版直播回放并下载演讲文档。一、洞察:房企经营思路的变化企业的转型都是围绕着业务经营变化进行的,房企数…...

零入门kubernetes网络实战-14->基于veth pair、namespace以及路由技术,实现跨主机命名空间之间的通信测试案例
《零入门kubernetes网络实战》视频专栏地址 https://www.ixigua.com/7193641905282875942 本篇文章视频地址(稍后上传) 本篇文章继续提供测试案例: 基于veth pair、namespace以及路由技术,实现跨主机命名空间之间的通信 1、网络拓扑如下 2、网络拓扑构建…...

【pytorch框架】对模型知识的基本了解
文章目录TensorBoard的使用1、TensorBoard启动:2、使用TensorBoard查看一张图片3、transforms的使用pytorch框架基础知识1 nn.module的使用2 nn.conv2d的使用3、池化(MaxPool2d)4 非线性激活5 线性层6 Sequential的使用7 损失函数与反向传播8 优化器9 对现有网络的使…...

SUP桨板电动气泵方案——鼎盛合方案
SUP桨板是现时最热门的水上运动之一,它的全称是Stand Up Paddle,简称SUP。这项运动近几年在我国三亚等地区风靡一时,在网上经常看到一些运动博主或者明星网红晒出冲浪视频,刺激又惊险。SUP桨板为充气式桨板,需要通过充…...

小白系列Vite-Vue3-TypeScript:011-登录界面搭建及动态路由配置
前面几篇文章我们介绍的都是ViteVue3TypeScript项目中环境相关的配置,接下来我们开始进入系统搭建部分。本篇我们来介绍登录界面搭建及动态路由配置,大家一起撸起来......搭建登录界面登陆接口api项目登陆接口是通过mockjs前端来模拟的模拟服务接口Login…...

C语言( 缓冲区和重定向)
一.缓冲输入,无缓存输入 while((chgetchar()) ! #) putchar(ch); 这里getchar(),putchar()每次只处理一个字符(这里只是知道就好了),而我们使用while循环,当读到#字符时停止 而看到输出例子,第一行我们输入…...

编程思想、方法论和架构的类型及应用
概要编程思想是指在编写代码时所采用的基本思维方式和方法论。分类编程思想编程思想为软件开发提供了思维范式和指导思路,例如面向对象思想、函数式编程思想等,它们帮助程序员更好地抽象问题、组织代码、提高代码复用性和可维护性,包括一下几…...

【OA办公】OA流程审批大揭秘,带你看遍所有基础流程
流程审批,是所有企业的OA办公系统重要组成部分,是任何OA办公系统都不可缺少的。比起传统的纸张传阅、签批的审批模式浪费了大量的时间和成本,因此越来越多的企业采用OA这种全新的、高效的、智能的审批模式。流程审批除了这些好处,…...

《零基础入门数据结构与算法》专栏介绍
目录 前言 第一部分:重点 第二部分:题库 第三部分:测试 第四部分:实验 第五部分:试卷 总结 前言 本专栏主要分为五个部分: ① 重要知识点详解 ② 近百道练习题解析 ③ 数据结构与算法章节测试 …...

测试开发之Django实战示例 第九章 扩展商店功能
第九章 扩展商店功能在上一章里,为电商站点集成了支付功能,然后可以生成PDF发票发送给用户。在本章,我们将为商店添加优惠码功能。此外,还会学习国际化和本地化的设置和建立一个推荐商品的系统。本章涵盖如下要点:建立…...

【Spring】一文带你吃透AOP面向切面编程技术(下篇)
个人主页: 几分醉意的CSDN博客_传送门 上节我们介绍了什么是AOP、Aspectj框架的前置通知Before传送门,这篇文章将继续详解Aspectj框架的其它注解。 文章目录💖Aspectj框架介绍✨JoinPoint通知方法的参数✨后置通知AfterReturning✨环绕通知Ar…...

【java】Spring Boot --40 个 Spring Boot 常用注解(建议收藏)
本文目录一、Spring Web MVC 注解Spring Web MVC 注解RequestMappingRequestBodyGetMappingPostMappingPutMappingDeleteMappingPatchMappingControllerAdviceResponseBodyExceptionHandlerResponseStatusPathVariableRequestParamControllerRestControllerModelAttributeCross…...

《游戏学习》| 微信对话模拟生成器源码分析
简介微信对话生成器,是一款在线微信聊天对话制作的工具,它可以设置苹果或安卓状态栏,包括手机电量、手机时间等,还可以设置不同用户的角色,然后发送文字、语音、红包、转账等多种好玩的功能,可谓是一款娱乐…...

剑指 Offer 10- I. 斐波那契数列[c语言]
目录题目思路代码结果该文章只是用于记录考研复试刷题题目 力扣斐波那契数列 写一个函数,输入 n ,求斐波那契(Fibonacci)数列的第 n 项(即 F(N))。斐波那契数列的定义如下: F(0) 0, F(1) 1 …...

【C#基础】C# 数据类型总结
序号系列文章0【C#基础】初识编程语言C#1【C#基础】C# 程序通用结构2【C#基础】C# 基础语法解析文章目录前言数据类型一. 值类型(Value types)二. 引用类型(Reference types)三. 指针类型(Pointer types)结…...

再创荣誉 | Softing工业荣获CAIMRS 2023 数字化创新奖
在刚刚结束的中国工控-第二十一届“自动化及数字化”年度评选(CAIMRS 2023)中,Softing凭借edgeAggregator产品荣获“数字化创新奖”! 经层层筛选,Softing edgeAggregator边缘聚合服务器从中脱颖而出,摘得C…...

Multi Paxos
basic paxos 是用于确定且只能确定一个值,“只确定一个值有什么用?这可解决不了我面临的问题,例如每个用户都要多次保存数据.” 你心中可能有这样的疑问。 原simple paxos论文里有提到一连串个instance of paxos [4] 但没有提出 multi paxos的概念&…...

Android - dimen适配
一、分辨率对应DPIDPI名称范围值分辨率名称屏幕分辨率density密度(1dp显示多少px)ldpi120QVGA240*3200.75(120dpi/1600.75px)mdpi160(基线)HVGA320*4801(160dpi/1601px)hdpi240WVGA4…...

深度学习网络模型——RepVGG网络详解
深度学习网络模型——RepVGG网络详解0 前言1 RepVGG Block详解2 结构重参数化2.1 融合Conv2d和BN2.2 Conv2dBN融合实验(Pytorch)2.3 将1x1卷积转换成3x3卷积2.4 将BN转换成3x3卷积2.5 多分支融合2.6 结构重参数化实验(Pytorch)3 模型配置论文名称: RepVGG: Making V…...

仓库拣货标签应用案例
使用场景:富士康成都仓库 解决问题:仓库亮灯拣选, 提高作业效率和物料明晰展示仓库亮灯拣选使用场景:京东仓库 解决问题:播种墙分拣,合单拣货完成后按订单播种播种墙分拣使用场景:和尔泰智能料…...