Nacos配置中心搭建及動(dòng)態(tài)刷新配置及踩坑記錄
近期有用到阿里的開(kāi)源配置中心及注冊(cè)中心nacos,特此記錄并分享一些學(xué)習(xí)筆記及配置踩坑點(diǎn)。
一、Nacos配置中心介紹
從架構(gòu)圖上可以知道,Nacos提供了兩種服務(wù),一種是用于服務(wù)注冊(cè)、發(fā)現(xiàn)的Naming Service,一種是用于配置中心、動(dòng)態(tài)配置的Config Service,而他們底層均由core模塊來(lái)支持。
外層提供OpenAPI供客戶端使用,并提供了User Console、Admin Console方便用戶使用 。

用戶通過(guò)管理平臺(tái)發(fā)布配置,通過(guò)HTTP調(diào)用將配置注冊(cè)到服務(wù)端,服務(wù)端將之保存在MySQL等持久化存儲(chǔ)引擎中;用戶通過(guò)客戶端SDK訪問(wèn)服務(wù)端的配置,同時(shí)建立HTTP的長(zhǎng)輪詢監(jiān)聽(tīng)配置項(xiàng)變更,同時(shí)為了減輕服務(wù)端壓力和保證容災(zāi)特性,配置項(xiàng)拉取到客戶端之后會(huì)保存一份快照在本地文件中,SDK優(yōu)先讀取本地文件里的配置。
當(dāng)服務(wù)端的配置發(fā)生變更時(shí),客戶端會(huì)通過(guò)監(jiān)聽(tīng)機(jī)制,拿到變更后的最新配置信息。

簡(jiǎn)單來(lái)說(shuō),Nacos 客戶端是怎么實(shí)時(shí)獲取到 Nacos 服務(wù)端的最新數(shù)據(jù)的:
Nacos 并不是通過(guò)推的方式將服務(wù)端最新的配置信息發(fā)送給客戶端的,而是客戶端維護(hù)了一個(gè)長(zhǎng)輪詢的任務(wù),定時(shí)去拉取發(fā)生變更的配置信息,然后將最新的數(shù)據(jù)推送給 Listener 的持有者。
二、配置中心搭建
1.配置中心啟用
首先需要到官網(wǎng)上下載nacos-server,然后將服務(wù)跑起來(lái)之后。
根據(jù)自己的需要,建立幾個(gè)命名空間,當(dāng)然使用默認(rèn)的public的命名空間也是可以的,如圖所示。

2.創(chuàng)建配置文件
在配置列表中創(chuàng)建配置文件,其中配置文件命名方式需要注意。
在Nacos-Server中新建配置,其中Data ID它的定義規(guī)則是:${prefix}-${spring.profiles.active}.${file-extension}
prefix默認(rèn)為 spring.application.name 的值,也可以通過(guò)配置項(xiàng) spring.cloud.nacos.config.prefix 來(lái)配置。spring.profiles.active即為當(dāng)前環(huán)境對(duì)應(yīng)的 profile,可以通過(guò)配置項(xiàng) spring.profiles.active 來(lái)配置。file-exetension為配置內(nèi)容的數(shù)據(jù)格式,可以通過(guò)配置項(xiàng) spring.cloud.nacos.config.file-extension 來(lái)配置。目前只支持 properties 和 yaml 類型。
注意:當(dāng) spring.profiles.active 為空時(shí),對(duì)應(yīng)的連接符 - 也將不存在,dataId 的拼接格式變成 prefix.{prefix}.prefix.{file-extension}
在本文中,我的項(xiàng)目名稱為vaso-core,配置文件使用的是dev環(huán)境,因此我的配置文件如下:

其中的文件內(nèi)容就是正常的yaml格式的內(nèi)容填寫就行,這邊就不具體展示了。

3.工程配置
首先我們需要導(dǎo)入nacos-config的依賴(注:我這邊沒(méi)有使用nacos的注冊(cè)發(fā)現(xiàn),依舊沿用自身使用的eureka)。
其中依賴版本的對(duì)應(yīng)關(guān)系可以在官網(wǎng)中查找。
<dependency>
<groupId>com.alibaba.cloud</groupId>
<artifactId>spring-cloud-starter-alibaba-nacos-config</artifactId>
<version>2.1.2.RELEASE</version>
</dependency>其次我們需要在配置文件中進(jìn)行配置,采用bootstrap.yml文件進(jìn)行配置,因?yàn)閟pringboot會(huì)優(yōu)先啟動(dòng)bootstrap.yml文件中的配置。其中文件內(nèi)容如下:
bootstrasp.yml:
spring:
application:
name: vaso-core
cloud:
nacos:
# discovery:
# server-addr: xx.xx.xxx.xx:8848
# namespace: 8b387ec4-e60d-4878-baf4-f2246e58a1d6
config:
server-addr: xx.xx.xxx.xx:8848
file-extension: yml
namespace: 8b387ec4-e60d-4878-baf4-f2246e58a1d6
profiles:
active: devnamespace的內(nèi)容為上文中配置的dev命名空間的id。
最后我們需要引入@refreshscope關(guān)鍵字,保證可以實(shí)時(shí)刷新配置。在controller層或者要刷新的值所在的類上進(jìn)行注釋就行。

其中我們的demo是如下所示:

三、試驗(yàn)結(jié)果
將工程跑起來(lái),可以看到一開(kāi)始就已經(jīng)去配置中心讀取配置了,如圖:

此時(shí)將配置中心內(nèi)容修改,可以看到已經(jīng)實(shí)時(shí)刷新了配置。

四、踩坑經(jīng)歷
其實(shí)在正式使用之前也碰到過(guò)nacos無(wú)法試試實(shí)時(shí)刷新的問(wèn)題,打斷點(diǎn)卻發(fā)現(xiàn)實(shí)際已經(jīng)讀取到本地,通過(guò)進(jìn)入spring的源碼中查看,也發(fā)現(xiàn)其實(shí)已經(jīng)可以拿到配置了,但是卻一直無(wú)法刷新配置。
最后通過(guò)一個(gè)禮拜的定位源碼,斷點(diǎn)查詢才發(fā)現(xiàn)一個(gè)本質(zhì)性的問(wèn)題。欲哭無(wú)淚
@RefreshScope與數(shù)據(jù)源之間存在沖突?。?/p>
具體如下:
- 因?yàn)轫?xiàng)目是多數(shù)據(jù)源,所以使用的是自定義數(shù)據(jù)源配置的DataSource,用@Bean注入。
- 在SpringBoot 2.0以上默認(rèn)使用Hikari連接池,一旦連接池啟動(dòng),就無(wú)法再修改HikariDataSource,所以刷新配置時(shí)連帶數(shù)據(jù)源一起刷新,于是會(huì)報(bào)錯(cuò)。
Caused by: java.lang.IllegalStateException: The configuration of the pool is sealed once started. Use HikariConfigMXBean for runtime changes.
解決方法: 在自定義的DataSource上加入注解@RefreshScope,或者使用spring.scloud.refresh.extra-refreshable配置指定classname列表即可。
查看Hikari的默認(rèn)配置即可發(fā)現(xiàn),這個(gè)與刷新配置之間是存在沖突的。

因此在數(shù)據(jù)源配置類中加入注釋,這樣每次刷新配置的時(shí)候會(huì)重新刷新這個(gè)Bean,那么就可以保證不會(huì)報(bào)錯(cuò),這樣的spring的刷新機(jī)制就可以順利執(zhí)行下去了。

五、總結(jié)
通過(guò)這次經(jīng)歷,我發(fā)現(xiàn)定位問(wèn)題的過(guò)程是一門極高的學(xué)問(wèn),學(xué)會(huì)怎么去看源碼,懂原理才能更好的更精準(zhǔn)的分析問(wèn)題所在之處。
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
SpringBoot使用MapStruct生成映射代碼的示例詳解
MapStruct 是一個(gè)用于 Java 的代碼生成器,專門用于生成類型安全的 bean 映射代碼,它通過(guò)注解處理器在編譯時(shí)生成映射代碼,從而避免了運(yùn)行時(shí)的性能開(kāi)銷和潛在的錯(cuò)誤,本文給大家介紹了SpringBoot使用MapStruct生成映射代碼的示例,需要的朋友可以參考下2024-11-11
java.nio.file.WatchService?實(shí)時(shí)監(jiān)控文件變化的示例代碼
在?Java?語(yǔ)言中,從?JDK7?開(kāi)始,新增了java.nio.file.WatchService類,用來(lái)實(shí)時(shí)監(jiān)控文件的變化,這篇文章主要介紹了java.nio.file.WatchService?實(shí)時(shí)監(jiān)控文件變化,需要的朋友可以參考下2022-05-05
Java的Struts框架中<results>標(biāo)簽的使用方法
這篇文章主要介紹了Java的Struts框架中<results>標(biāo)簽的使用方法,Struts框架是Java的SSH三大web開(kāi)發(fā)框架之一,需要的朋友可以參考下2015-11-11
eclipse創(chuàng)建java項(xiàng)目并運(yùn)行的詳細(xì)教程講解
eclipse是java開(kāi)發(fā)的ide工具,是大部分java開(kāi)發(fā)人員的首選開(kāi)發(fā)工具,可是對(duì)于一些新Java人員來(lái)說(shuō),不清楚eclipse怎么運(yùn)行項(xiàng)目?這篇文章主要給大家介紹了關(guān)于eclipse創(chuàng)建java項(xiàng)目并運(yùn)行的相關(guān)資料,需要的朋友可以參考下2023-04-04
淺析Java中JNI靜態(tài)注冊(cè)和動(dòng)態(tài)注冊(cè)
這篇文章主要為大家詳細(xì)介紹了Java中JNI靜態(tài)注冊(cè)和動(dòng)態(tài)注冊(cè)的相關(guān)知識(shí),文中的示例代碼簡(jiǎn)潔易懂,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-12-12
Spring?Boot如何通過(guò)Actuator顯示git和build的信息
這篇文章主要介紹了Spring?Boot通過(guò)Actuator顯示git和build的信息,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-01-01

