SpringCloud 服務註冊IP錯誤的解決

SpringCloud 服務註冊IP錯誤

1、錯誤原因

在服務註冊的時候,是使用 spring.cloud.client.ipAddress 這個變量,如果本機有多個網卡,那麼可能會把不是本機以太網的網卡地址註冊上去。

使用 ipconfig 可以看到,本機上有多個以太網適配器,而每個以太網適配器,都有一個 IPv4 地址,這時註冊上去的 IP,就是其中一個,卻不一定是正確的那個。

2、處理

2.1、禁用其他網卡

到電腦的 更改適配器 設置中,將不是本機以太網的其他網卡禁用

2.2、配置

到電腦的設備管理器 –> 網絡適配器 中,可以看到所有的網卡名

在要註冊的服務中配置一下內容:

//忽略指定正則匹配的網卡的配置,我這裡配置瞭VM虛擬機和Docker的
spring.cloud.inetutils.ignoredInterfaces=['VMware.*','Hyper-V.*']
//指定默認IP,可以使IP段
spring.cloud.inetutils.preferredNetworks=['192.168']
spring.cloud.inetutils.use-only-site-local-interfaces=true

SpringCloud以及Nacos服務註冊IP選擇

微服務部署後,需要相互調用,其中服務A調用服務B時發現無法調用。其中服務註冊和發現以及配置中心使用Nacos

分析:

檢查瞭多遍代碼後,沒有發現調用方式有問題,所以隻能是網絡問題。通過postman直接調用服務B,發現可以調通,但是使用服務A不行,於是檢查服務A在註冊中心註冊的IP,發現和並不是服務B啟動機器的IP。這就是問題所在瞭。

為什麼註冊的IP和真實IP不符合呢?原因是Nacos客戶端在註冊服務時會從機器網卡中選擇一個IP來註冊,當機器存在多個網卡(例如存在虛擬網卡)時,所選則的IP可能不是真是的物理機的IP,所以,當註冊瞭的是非真實IP後,另一臺機器調用時是不可能調通的。

解決:

知道問題後,就要解決,查瞭一下SpringCloud的官方文檔,發現有一項配置如下:

Sometimes, it is useful to ignore certain named network interfaces so that they can be excluded from Service Discovery registration (for example, when running in a Docker container).

A list of regular expressions can be set to cause the desired network interfaces to be ignored.

You can also force the use of only specified network addresses by using a list of regular expressions.

spring:
  cloud:
 inetutils:
   preferredNetworks:
  - 192.168
  - 10.0

該項配置用於指定首選IP,當有多個網卡時,指定該IP地址後(支持正則),客戶端在選擇IP時就會選擇符合preferredNetworks配置的IP地址進行註冊。

同樣的,Nacos也可以配置自己的首選IP以及網卡選擇:

spring.cloud.nacos.discovery.ip:
spring.cloud.nacos.discovery.networkInterface

我們選擇其中一個配置就可以,都能達到相同的效果。

擴展:

雖然問題解決瞭,但是還是要更深入的瞭解一下這個IP選擇的邏輯。翻瞭一通源碼發現,其大致邏輯如下:

Nacos首先檢查有沒有ip及networkInterface配置,如果有則使用配置的IP,否則檢查networkInterface,並獲取IP,如果兩者都為空,則使用inetUtils.findFirstNonLoopbackHostInfo().getIpAddress()來獲取IP:

而findFirstNonLoopbackHostInfo()的部分邏輯如下:

它最終會返回一個匹配的IPV4地址,並且排除本機回環網絡(127.0.0.0-127.255.255.255),並且匹配是否是首選網絡(如果配置瞭preferredNetworks)。

以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。

推薦閱讀: