色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

Nacos+@RefreshScope為什么配置能動態刷新?

jf_ro2CN3Fa ? 來源:JAVA旭陽 ? 2023-05-19 14:15 ? 次閱讀

概述

RefeshScope這個注解想必大家都用過,在微服務配置中心的場景下經常出現,他可以用來刷新Bean中的屬性配置,那大家對他的實現原理了解嗎?它為什么可以做到動態刷新呢?

注解的作用

@RefreshScope注解是Spring Cloud中的一個注解,用來實現Bean中屬性的動態刷新。

/**
*Convenienceannotationtoputa@Beandefinitionin
*{@linkorg.springframework.cloud.context.scope.refresh.RefreshScoperefreshscope}.
*Beansannotatedthiswaycanberefreshedatruntimeandanycomponentsthatareusing
*themwillgetanewinstanceonthenextmethodcall,fullyinitializedandinjected
*withalldependencies.
*
*@authorDaveSyer
*
*/
@Target({ElementType.TYPE,ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Scope("refresh")
@Documented
public@interfaceRefreshScope{

/**
*@seeScope#proxyMode()
*@returnproxymode
*/
ScopedProxyModeproxyMode()defaultScopedProxyMode.TARGET_CLASS;

}

上面是RefreshScope的源碼,該注解被@Scope注解使用,@Scope用來比較Spring Bean的作用域,具體使用參考相關文章。

注解的屬性proxyMode默認使用TARGET_CLASS作為代理。

實例

controller中添加@RefreshScope

1d47d002-f5fc-11ed-90ce-dac502259ad0.png

nacos配置中心中配置

1d6e088a-f5fc-11ed-90ce-dac502259ad0.png

驗證, 修改配置中心后,可以不重啟動,刷新配置

1d839e70-f5fc-11ed-90ce-dac502259ad0.png1d9adefa-f5fc-11ed-90ce-dac502259ad0.png1db9bad2-f5fc-11ed-90ce-dac502259ad0.png

去掉@RefreshScope 就不會自動刷新。

原理解析

為了實現動態刷新配置,主要就是想辦法達成以下兩個核心目標:

讓Spring容器重新加載Environment環境配置變量

Spring Bean重新創建生成

@RefreshScope主要就是基于@Scope注解的作用域代理的基礎上進行擴展實現的,加了@RefreshScope注解的類,在被Bean工廠創建后會加入自己的refresh scope 這個Bean緩存中,后續會優先從Bean緩存中獲取,當配置中心發生了變更,會把變更的配置更新到spring容器的Environment中,并且同事bean緩存就會被清空,從而就會從bean工廠中創建bean實例了,而這次創建bean實例的時候就會繼續經歷這個bean的生命周期,使得@Value屬性值能夠從Environment中獲取到最新的屬性值,這樣整個過程就達到了動態刷新配置的效果。

1dd650f2-f5fc-11ed-90ce-dac502259ad0.png

獲取RefreshScope注解的Bean

1e5a8502-f5fc-11ed-90ce-dac502259ad0.png

通過打上斷點查看堆棧可知:

因為Class被加上了@RefreshScope注解,那么這個BeanDefinition信息中的scope為refresh,在getBean的的時候會單獨處理邏輯。

publicabstractclassAbstractBeanFactoryextendsFactoryBeanRegistrySupportimplementsConfigurableBeanFactory{

protectedTdoGetBean(
Stringname,@NullableClassrequiredType,@NullableObject[]args,booleantypeCheckOnly)
throwsBeansException{

//如果scope是單例的情況,這里不進行分析
if(mbd.isSingleton()){
.....
}
//如果scope是prototype的情況,這里不進行分析
elseif(mbd.isPrototype()){
......
}
//如果scope是其他的情況,本例中是reresh
else{
StringscopeName=mbd.getScope();
if(!StringUtils.hasLength(scopeName)){
thrownewIllegalStateException("Noscopenamedefinedforbean'"+beanName+"'");
}
//獲取refreshscope的實現類RefreshScope,這個類在哪里注入,我們后面講
Scopescope=this.scopes.get(scopeName);
if(scope==null){
thrownewIllegalStateException("NoScoperegisteredforscopename'"+scopeName+"'");
}
try{
//這邊是獲取bean,調用的是RefreshScope中的的方法
ObjectscopedInstance=scope.get(beanName,()->{
beforePrototypeCreation(beanName);
try{
returncreateBean(beanName,mbd,args);
}
finally{
afterPrototypeCreation(beanName);
}
});
beanInstance=getObjectForBeanInstance(scopedInstance,name,beanName,mbd);
}
catch(IllegalStateExceptionex){
thrownewScopeNotActiveException(beanName,scopeName,ex);
}
}
}
catch(BeansExceptionex){
beanCreation.tag("exception",ex.getClass().toString());
beanCreation.tag("message",String.valueOf(ex.getMessage()));
cleanupAfterBeanCreationFailure(beanName);
throwex;
}
finally{
beanCreation.end();
}
}

returnadaptBeanInstance(name,beanInstance,requiredType);
}

}

2.RefreshScope繼承成了GenericScope類,最終調用的的是GenericScope的get方法

publicclassGenericScope
implementsScope,BeanFactoryPostProcessor,BeanDefinitionRegistryPostProcessor,DisposableBean{
@Override

publicObjectget(Stringname,ObjectFactoryobjectFactory){
//將bean添加到緩存cache中
BeanLifecycleWrappervalue=this.cache.put(name,newBeanLifecycleWrapper(name,objectFactory));
this.locks.putIfAbsent(name,newReentrantReadWriteLock());
try{
//調用下面的getBean方法
returnvalue.getBean();
}
catch(RuntimeExceptione){
this.errors.put(name,e);
throwe;
}
}

privatestaticclassBeanLifecycleWrapper{

publicObjectgetBean(){
//如果bean為空,則創建bean
if(this.bean==null){
synchronized(this.name){
if(this.bean==null){
this.bean=this.objectFactory.getObject();
}
}
}
//否則返回之前創建好的bean
returnthis.bean;
}
}
}

小結:

從這邊的代碼中可以印證了上面的說法,創建后的Bean會緩存到scope的cache中,優先從緩存中獲取,如果緩存中是null, 則重新走一遍create bean的流程。

RefeshScope Bean的創建

上面的在getBean的時候依賴到RefreshScope這個Bean,那么這個Bean是在什么時候加入到Spring Bean中的呢?答案就是RefreshAutoConfiguration。

1e98f7b0-f5fc-11ed-90ce-dac502259ad0.png

配置中心刷新后刷新Bean緩存

1ec8ccd8-f5fc-11ed-90ce-dac502259ad0.png

配置中心發生變化后,會收到一個RefreshEvent事件,RefreshEventListner監聽器會監聽到這個事件。

publicclassRefreshEventListenerimplementsSmartApplicationListener{


........

publicvoidhandle(RefreshEventevent){
if(this.ready.get()){//don'thandleeventsbeforeappisready
log.debug("Eventreceived"+event.getEventDesc());
//會調用refresh方法,進行刷新
Setkeys=this.refresh.refresh();
log.info("Refreshkeyschanged:"+keys);
}
}

}

//這個是ContextRefresher類中的刷新方法
publicsynchronizedSetrefresh(){
//刷新spring的envirionment變量配置
Setkeys=refreshEnvironment();
//刷新其他scope
this.scope.refreshAll();
returnkeys;
}

refresh方法最終調用destroy方法,清空之前緩存的bean

publicclassRefreshScopeextendsGenericScope
implementsApplicationContextAware,ApplicationListener,Ordered{

@ManagedOperation(description="Disposeofthecurrentinstanceofallbeans"
+"inthisscopeandforcearefreshonnextmethodexecution.")
publicvoidrefreshAll(){
//調用父類的destroy
super.destroy();
this.context.publishEvent(newRefreshScopeRefreshedEvent());
}
}


@Override
publicvoiddestroy(){
Listerrors=newArrayList();
Collectionwrappers=this.cache.clear();
for(BeanLifecycleWrapperwrapper:wrappers){
try{
Locklock=this.locks.get(wrapper.getName()).writeLock();
lock.lock();
try{
//這里主要就是把之前的bean設置為null,就會重新走createBean的流程了
wrapper.destroy();
}
finally{
lock.unlock();
}
}
catch(RuntimeExceptione){
errors.add(e);
}
}
if(!errors.isEmpty()){
throwwrapIfNecessary(errors.get(0));
}
this.errors.clear();
}

總結

上面是這個RefreshScope實現動態刷新大致的原理,其中里面還有很多細節,可能需要留給大家自己debug去深入理解。





審核編輯:劉清

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • cache技術
    +關注

    關注

    0

    文章

    41

    瀏覽量

    1069
  • null
    +關注

    關注

    0

    文章

    19

    瀏覽量

    3985

原文標題:Nacos+@RefreshScope 為什么配置能動態刷新?

文章出處:【微信號:芋道源碼,微信公眾號:芋道源碼】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    Nacos是什么?Nacos配置管理技巧你知道嗎

    Nacos 是阿里巴巴今年7月份開源的項目,如其名, Naming Configuration Service ,專注于服務發現和配置管理領域。本系列文章,將從 5W1H(What、Where
    的頭像 發表于 10-29 08:53 ?1.5w次閱讀

    Nacos的概念和功能

    1、Nacos簡介 Nacos的概念和功能 Nacos是一個面向微服務架構的動態服務發現、配置管理和服務治理平臺,它能夠幫助開發人員和運維人
    的頭像 發表于 09-25 11:02 ?2362次閱讀

    支持Dubbo生態發展,阿里巴巴啟動新的開源項目 Nacos

    ,其核心定位是 “一個更易于幫助構建云原生應用的動態服務發現、配置和服務管理平臺”。Nacos 有三大主要功能:服務發現與服務管理在采用以“服務(Service)”為中心的諸如微服務及云原生方式的現代
    發表于 07-05 17:35

    結合場景談一談微服務配置

    ,最終是抽象為一個個的配置項,要想實現運行時的動態調整閾值和開關的啟停,將這些配置項存放到 Nacos配置模塊中最適合不過了。在今年 8
    發表于 12-12 15:53

    構建ARM64版本nacos docker鏡像

    在適配過程中有大量合作伙伴用到nacos且采用容器化部署,dockerhub未提供官方鏡像,因此需要在鯤鵬服務器自定義構建。構建前提:Docker已部署構建步驟:1、下載包含構建所需的腳本下載完成
    發表于 06-16 14:29

    基于71M6515H和雙CPU的智能動態分相無功補償控制器

    基于71M6515H和雙CPU的智能動態分相無功補償控制器摘要:為了改善電能質量,設計了一種基于71M6515H 和雙CPU 的智能動態分相無功補償控制器,利用71M6515H 解決了
    發表于 05-12 16:10 ?32次下載

    基于LabView平臺的齒輪箱性能動態測試與診斷_李貴明

    基于LabView平臺的齒輪箱性能動態測試與診斷_李貴明
    發表于 03-18 09:41 ?3次下載

    Nacos發布0.5.0版本,輕松玩轉動態 DNS 服務

    Gateway 動態路由配置,等到這塊足夠成熟,會將其包含在Nacos的官方推薦實現中。四、TTL & Health Status Aggregation在Nacos之前的幾個版本中
    發表于 12-05 16:22 ?120次閱讀

    微服務配置中心實戰:Spring + MyBatis + Druid + Nacos

    在 結合場景談服務發現和配置 中我們講述了 Nacos 配置中心的三個典型的應用場景,包括如何在 Spring Boot 中使用 Nacos 配置
    發表于 12-29 17:09 ?1115次閱讀
    微服務<b class='flag-5'>配置</b>中心實戰:Spring + MyBatis + Druid + <b class='flag-5'>Nacos</b>

    Nacos服務地址動態感知原理

    Nacos Server:Nacos服務提供者,里面包含的Open API是功能訪問入口,Conig Service、Naming Service 是Nacos提供的配置服務、命名服務
    的頭像 發表于 09-26 10:40 ?1780次閱讀

    Nacos為什么這么強?Nacos注冊中心的底層原理,從服務注冊到服務發現

    來源:碼猿技術專欄 1. Nacos介紹 2. Nacos注冊中心實現原理分析 2.1 Nacos架構圖 2.2 注冊中心的原理 3. Nacos源碼分析 3.1
    的頭像 發表于 10-08 16:46 ?1.2w次閱讀

    Nacos、Apollo、Config配置中心如何選型?

    傳統的靜態配置方式要想修改某個配置只能修改之后重新發布應用,要實現動態性,可以選擇使用數據庫,通過定時輪詢訪問數據庫來感知配置的變化。輪詢頻率低感知
    的頭像 發表于 10-31 11:14 ?1431次閱讀

    華為云CSE 關鍵特性,支持托管Nacos注冊配置中心

    于構建云原生應用的動態服務發現、配置管理和服務管理平臺。 Nacos 幫助您發現、配置和管理微服務。Nacos?提供了一組簡單易用的特性集,
    的頭像 發表于 12-29 16:23 ?1046次閱讀
    華為云CSE 關鍵特性,支持托管<b class='flag-5'>Nacos</b>注冊<b class='flag-5'>配置</b>中心

    基于Nacos的簡單動態化線程池實現

    本文以Nacos作為服務配置中心,以修改線程池核心線程數、最大線程數為例,實現一個簡單的動態化線程池。
    發表于 01-06 14:14 ?869次閱讀

    Nacos實現原理:SpringCloud集成Nacos的實現過程

    Nacos服務提供者,里面包含的Open API是功能訪問入口,Conig Service、Naming Service 是Nacos提供的配置服務、命名服務模塊。Consitency
    發表于 10-09 16:08 ?1122次閱讀
    <b class='flag-5'>Nacos</b>實現原理:SpringCloud集成<b class='flag-5'>Nacos</b>的實現過程
    主站蜘蛛池模板: 美女脱内衣裸身尿口露出来| 国自产精品手机在线视频| 吻嘴胸全身好爽床大全| 欧美精品做人一级爱免费| 久久免费国产| 久久成人精品免费播放| 韩剧甜性涩爱| 国产专区亚洲欧美另类在线| 国产精品久久久久久久久免费下载 | 97公开超碰在线视频| 伊人久久大香线蕉综合网站| 亚洲视频免费在线观看| 亚洲国产精品高清在线| 推倒美女总裁啪啪| 天堂草原天黑黑| 无码AV精品一区二区三区| 香蕉动漫库| 亚洲毛片网| 曰本女人牲交视频免费| 中字幕久久久人妻熟女天美传媒 | Y8848高清私人影院软件优势| 99这里只有是精品2| sihu国产精品永久免费| 菠萝菠萝蜜在线观看视频| 川师 最美老师| 国产精品久人妻精品| 国产亚洲人成在线视频| 久久re这里视频精品8| 久久无码人妻AV精品一区| 免费女人光着全身网站| 欧美夜夜噜2017最新| 骚浪插深些好烫喷了| 性夜夜春夜夜爽AA片A| 亚洲一区二区三区91| 4399亚洲AV无码V无码网站| G国产精品无马| 国产第一页在线视频| 好看AV中文字幕在线观看| 久久午夜伦理| 欧美日韩高清一区二区三区| 色欲色香天天天综合|