同時(shí)使用@LoadBalanced?@RefreshScope注解負(fù)載均衡失效分析
背景
最近引入了 Nacos Config 配置管理能力,說起來用法很簡(jiǎn)單,還是踩了三個(gè)坑。
- Nacos Config 的 nacos 的帳號(hào)密碼加密配置后,怎么解密而且在
NacosConfigBootstrapConfiguration
真正注入 Nacos Config 注入之前,而且不能觸發(fā)NacosDiscoveryProperties
的isNacosDiscoveryInfoChanged
變動(dòng)事件。因?yàn)?NacosDiscoveryProperties
接受NacosContextRefresher
事件時(shí),還是從 yml 配置中獲取屬性,不會(huì)從Environment
對(duì)象中加載。 @RefreshScope
要想生效,該注意什么?非 shared-config 的配置變動(dòng)時(shí),要想實(shí)時(shí)生效,必須在當(dāng)前應(yīng)用的 bootstrap.yml 中配置spring.application.name
屬性,注冊(cè)該應(yīng)用在配置中心需要監(jiān)聽的配置。@RefreshScope + @LoadBalanced
同時(shí)使用導(dǎo)致 Ribbon 負(fù)載均衡失效問題。
問題一比較復(fù)雜,此處不做討論,本文記錄問題三的解決方法及個(gè)人思考。
問題
有個(gè)模塊使用了 @LoadBalanced
負(fù)載均衡,通過配置控制超時(shí)時(shí)間。
引入 Nacos Config 配置后,按照常規(guī)用法,在對(duì)象上添加了 @RefreshScope
屬性,希望配置變動(dòng)時(shí),能實(shí)時(shí)生效。
注入代碼如下:
@Value("${rest.template.connect-timeout:10000}") private Integer connectTimeout; @Value("${rest.template.read-timeout:10000}") private Integer readTimeout; @Bean @RefreshScope @LoadBalanced public RestTemplate restTemplate(RestTemplateBuilder builder){ SimpleClientHttpRequestFactory requestFactory = new SimpleClientHttpRequestFactory(); requestFactory.setConnectTimeout(connectTimeout); //連接超時(shí)設(shè)置 requestFactory.setReadTimeout(readTimeout); //讀寫超時(shí)設(shè)置 RestTemplate restTemplate = new RestTemplate(requestFactory); logger.info("初始化負(fù)載均衡的 RestTemplate 對(duì)象 {} {}", connectTimeout, readTimeout); return restTemplate; }
修改配置中心的配置后,可以監(jiān)控到 RestTemplate
會(huì)在配置變化后重新初始化了,也打印了最新的配置。
但真正使用這個(gè)類,調(diào)用某個(gè)服務(wù)時(shí),出現(xiàn)了服務(wù)無法解析的異常:
分析
首先,檢查調(diào)用的目標(biāo)服務(wù)是否注冊(cè)成功,目標(biāo)服務(wù)是正常啟動(dòng)的。
其次,對(duì)比其他同樣引用了 @LoadBalanced
的 RestTemplate
的模塊,它調(diào)用是正常的。
最后,對(duì)比異常調(diào)用和正常調(diào)用的注入代碼的區(qū)別,多了一個(gè) @RefreshScope
,調(diào)整代碼驗(yàn)證結(jié)果正常。引入 @RefreshScope
時(shí),在配置變化后才會(huì)發(fā)生這個(gè)異常,首次運(yùn)行時(shí)正常的。
延伸搜索發(fā)現(xiàn),負(fù)載均衡 RestTemplate
也有類似的問題 @scope("prototype")+@loadbalanced注解時(shí)負(fù)載均衡失效問題。
啟示錄
@LoadBalanced
和 @RefreshScope
同時(shí)使用,首次初始化時(shí),RestTemplate
對(duì)象具有負(fù)載均衡的能力;當(dāng) Nacos 配置中的配置變動(dòng)時(shí),這個(gè)對(duì)象會(huì)重新創(chuàng)建,而且此時(shí)并沒有使用 @LoadBalanced
的能力,導(dǎo)致負(fù)載均衡失效。
就是說,這兩個(gè)注解同時(shí)使用時(shí),在不同的時(shí)機(jī),只會(huì)有一個(gè)注解生效:
- 初始創(chuàng)建時(shí),
@LoadBalanced
生效,系統(tǒng)中的實(shí)例是負(fù)載均衡的RestTemplate
; - 當(dāng) Nacos 配置變化,
NacosContextRefresher
觸發(fā)通知@RefreshScope
注解的@Bean
對(duì)象時(shí),重新創(chuàng)建的實(shí)例就是普通的RestTemplate
了。
引入任何一個(gè)第三方工具,面對(duì)的都是黑盒,各種資料用法看似簡(jiǎn)單,一用就坑不斷??!建議 Nacos Config 官方配置給出一個(gè)使用建議 @RefreshScope
不要用在 @LoadBalanced
注解上。
最后記錄一個(gè)偶然的發(fā)現(xiàn):
選中幾個(gè)圖片文件后,右側(cè)概覽圖是一個(gè)堆疊的圖,才注意到!
以上就是同時(shí)使用@LoadBalanced @RefreshScope負(fù)載均衡失效分析的詳細(xì)內(nèi)容,更多關(guān)于@LoadBalanced @RefreshScope失效的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Springboot?異步任務(wù)和定時(shí)任務(wù)的異步處理
本文介紹了Springboot異步任務(wù)和定時(shí)任務(wù)的異步處理,Springboot?中,異步任務(wù)和定時(shí)任務(wù)是經(jīng)常遇到的處理問題方式,為了能夠用好這兩項(xiàng)配置,不干擾正常的業(yè)務(wù),需要對(duì)其進(jìn)行異步化配置。怎么設(shè)置合理的異步處理線程就是其核心和關(guān)鍵,下文詳情需要的朋友可以參考下2022-05-05java 全角半角字符轉(zhuǎn)換如何實(shí)現(xiàn)
在java中可能會(huì)用到過全角半角字符轉(zhuǎn)換問題,于是網(wǎng)上搜索整理了一下,曬出來和大家分享,希望可以幫助你們2012-12-12SpringCloud中的熔斷監(jiān)控HystrixDashboard和Turbine示例詳解
HystrixDashboard是用于實(shí)時(shí)監(jiān)控Hystrix性能的工具,展示請(qǐng)求響應(yīng)時(shí)間和成功率等數(shù)據(jù),本文介紹了如何配置和使用HystrixDashboard和Turbine進(jìn)行熔斷監(jiān)控,包括依賴添加、啟動(dòng)類配置和測(cè)試流程,感興趣的朋友一起看看吧2024-09-09SpringMVC @RequestMapping注解作用詳解
通過@RequestMapping注解可以定義不同的處理器映射規(guī)則,下面這篇文章主要給大家介紹了關(guān)于SpringMVC中@RequestMapping注解用法的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-01-01Java使用MySQL實(shí)現(xiàn)連接池代碼實(shí)例
這篇文章主要介紹了Java使用MySQL實(shí)現(xiàn)連接池代碼實(shí)例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-03-03