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

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

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

3天內不再提示

解析加載及實例化Bean的順序(零配置)

5jek_harmonyos ? 來源:CSDN ? 作者:低調的JVM ? 2021-08-04 16:08 ? 次閱讀

作者丨低調的JVM

來自丨CSDN

https://blog.csdn.net/qq_27529917/article/details/79329809

在使用Spring時,Bean之間會有些依賴,比如一個Bean A實例化時需要用到Bean B,那么B應該在A之前實例化好。很多時候Spring智能地為我們做好了這些工作,但某些情況下可能不是,比如Springboot的@AutoConfigureAfter注解,手動的指定Bean的實例化順序。

了解Spring內Bean的解析,加載和實例化順序機制有助于我們更好的使用Spring/Springboot,避免手動的去干預Bean的加載過程,搭建更優雅的框架。

Spring容器在實例化時會加載容器內所有非延遲加載的單例類型Bean,看如下源碼:

public abstract class AbstractApplicationContext extends DefaultResourceLoader

implements ConfigurableApplicationContext, DisposableBean {

//刷新Spring容器,相當于初始化

public void refresh() throws BeansException, IllegalStateException {

。。。。。。

// Instantiate all remaining (non-lazy-init) singletons.

finishBeanFactoryInitialization(beanFactory);

}

}

public class DefaultListableBeanFactory extends AbstractAutowireCapableBeanFactory

implements ConfigurableListableBeanFactory, BeanDefinitionRegistry, Serializable {

/** List of bean definition names, in registration order */

private volatile List《String》 beanDefinitionNames = new ArrayList《String》(256);

public void preInstantiateSingletons() throws BeansException {

List《String》 beanNames = new ArrayList《String》(this.beanDefinitionNames);

for (String beanName : beanNames) {

。。。。。。

getBean(beanName); //實例化Bean

}

}

}

ApplicationContext內置一個BeanFactory對象,作為實際的Bean工廠,和Bean相關業務都交給BeanFactory去處理。

在BeanFactory實例化所有非延遲加載的單例Bean時,遍歷beanDefinitionNames 集合,按順序實例化指定名稱的Bean。beanDefinitionNames 屬性是Spring在加載Bean Class生成的BeanDefinition時,為這些Bean預先定義好的名稱,看如下代碼:

public class DefaultListableBeanFactory extends AbstractAutowireCapableBeanFactory

implements ConfigurableListableBeanFactory, BeanDefinitionRegistry, Serializable {

public void registerBeanDefinition(String beanName, BeanDefinition beanDefinition)

throws BeanDefinitionStoreException {

。。。。。。

this.beanDefinitionNames.add(beanName);

}

}

BeanFactory在加載一個BeanDefinition(也就是加載Bean Class)時,將相應的beanName存入beanDefinitionNames屬性中,在加載完所有的BeanDefinition后,執行Bean實例化工作,此時會依據beanDefinitionNames的順序來有序實例化Bean,也就是說Spring容器內Bean的加載和實例化是有順序的,而且近似一致,當然僅是近似。

Spring在初始化容器時,會先解析和加載所有的Bean Class,如果符合要求則通過Class生成BeanDefinition,存入BeanFactory中,在加載完所有Bean Class后,開始有序的通過BeanDefinition實例化Bean。

我們先看加載Bean Class過程,零配置下Spring Bean的加載起始于ConfigurationClassPostProcessor的postProcessBeanDefinitionRegistry(BeanDefinitionRegistry)方法,我總結了下其加載解析Bean Class的流程:

配置類可以是Spring容器的起始配置類,也可以是通過@ComponentScan掃描得到的類,也可以是通過@Import引入的類。如果這個類上含有@Configuration,@Component,@ComponentScan,@Import,@ImportResource注解中的一個,或者內部含有@Bean標識的方法,那么這個類就是一個配置類,Spring就會按照一定流程去解析這個類上的信息

在解析的第一步會校驗當前類是否已經被解析過了,如果是,那么需要按照一定的規則處理(@ComponentScan得到的Bean能覆蓋@Import得到的Bean,@Bean定義的優先級最高)。

如果未解析過,那么開始解析:

解析內部類,查看內部類是否應該被定義成一個Bean,如果是,遞歸解析。

解析@PropertySource,也就是解析被引入的Properties文件。

解析配置類上是否有@ComponentScan注解,如果有則執行掃描動作,通過掃描得到的Bean Class會被立即解析成BeanDefinition,添加進beanDefinitionNames屬性中。之后查看掃描到的Bean Class是否是一個配置類(大部分情況是,因為標識@Component注解),如果是則遞歸解析這個Bean Class。

解析@Import引入的類,如果這個類是一個配置類,則遞歸解析。

解析@Bean標識的方法,此種形式定義的Bean Class不會被遞歸解析

解析父類上的@ComponentScan,@Import,@Bean,父類不會被再次實例化,因為其子類能夠做父類的工作,不需要額外的Bean了。

在1,3,4,6中都有遞歸操作,也就是在解析一個Bean Class A時,發現其上能夠獲取到其他Bean Class B信息,此時會遞歸的解析Bean Class B,在解析完Bean Class B后再接著解析Bean Class A,可能在解析B時能夠獲取到C,那么也會先解析C再解析B,就這樣不斷的遞歸解析。

在第3步中,通過@ComponentScan掃描直接得到的Bean Class會被立即加載入beanDefinitionNames中,但@Import和@Bean形式定義的Bean Class則不會,也就是說正常情況下面@ComponentScan直接得到的Bean其實例化時機比其他兩種形式的要早。

通過@Bean和@Import形式定義的Bean Class不會立即加載,他們會被放入一個ConfigurationClass類中,然后按照解析的順序有序排列,就是圖片上的 “將配置類有序排列”。一個ConfigurationClass代表一個配置類,這個類可能是被@ComponentScan掃描到的,則此類已經被加載過了;也可能是被@Import引入的,則此類還未被加載;此類中可能含有@Bean標識的方法。

Spring在解析完了所有Bean Class后,開始加載ConfigurationClass。如果這個ConfigurationClass是被Import的,也就是說在加載@ComponentScan時其未被加載,那么此時加載ConfigurationClass代表的Bean Class。然后加載ConfigurationClass內的@Bean方法。

順序總結:@ComponentScan 》 @Import 》 @Bean

Bean Class的結構圖如上所示,A是配置類的入口,通過A能直接或間接的引入一個模塊。

此時啟動Spring容器,將A引入容器內。

如果A是通過@ComponentScan掃描到的,那么此時的加載順序是:

A 》 D 》 F 》 B 》 E 》 G 》 C

如果A是通過@Import形式引入的,那么此時的加載順訊是:

D 》 F 》 B 》 E 》 G 》 A 》 C

當然以上僅僅代表著加載Bean Class的順序,實際實例化Bean的順序和加載順序大體相同,但還是會有一些差別。

Spring在通過getBean(beanName)形式實例化Bean時,會通過BeanDefinition去生成Bean對象。在這個過程中,如果BeanDefinition的DependsOn不為空,從字面理解就是依賴某個什么,其值一般是某個或多個beanName,也就是說依賴于其他Bean。

此時Spring會將DependsOn指定的這些名稱的Bean先實例化,也就是先調用getBean(dependsOn)方法。我們可以通過在Bean Class或者@Bean的方法上標識**@DependsOn**注解,來指定當前Bean實例化時需要觸發哪些Bean的提前實例化。

當一個Bean A內部通過@Autowired或者@Resource注入Bean B,那么在實例化A時會觸發B的提前實例化,此時會注冊A》B的dependsOn依賴關系,實質和@DependsOn一樣,這個是Spring自動為我們處理好的。

了解Spring Bean的解析,加載及實例化的順序機制能夠加深對Spring的理解,搭建更優雅簡介的Spring框架。

編輯:jq

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

    關注

    0

    文章

    340

    瀏覽量

    14338

原文標題:Spring解析,加載及實例化Bean的順序(零配置)

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

收藏 人收藏

    評論

    相關推薦

    SSM框架的源碼解析與理解

    的核心是控制反轉(IoC)和面向切面編程(AOP)。 源碼解析: Spring的源碼主要分為以下幾個部分: Bean容器: 負責實例配置
    的頭像 發表于 12-17 09:20 ?172次閱讀

    自動創建UI并解析數據

    *附件:32960_auto.rar備注:Main.vi是ui自動2.1.vi,配置文件為32960.B.ini。 目前可以實現根據配置文件自動創建控件并布局,且可以自動
    發表于 12-10 08:41

    DP83869HM使用100m光口模式時,需要配置哪些寄存器和配置順序呢?

    驅動,還是需要自己修改使其全部訪問間接地址(c00開始的) 使用100m光口模式時,需要配置哪些寄存器和配置順序呢?謝謝!
    發表于 12-05 07:10

    自動創建UI并解析數據

    Labview實現動態增加控件效果 - *附件:32960_auto.rar 備注:Main.vi是ui自動2.1.vi,配置文件為32960.B.ini。 目前可以實現根據配置文件自動
    發表于 11-29 11:26

    請問LMX2595配置時如何進行初始

    您好,查閱手冊說初始按照1.r0的rest置為1;2.r0的reset置為1;3.按照寄存器表格配置寄存器;4.fcal_en置為1的順序來初始。請問在第三步中,按照寄存器表
    發表于 11-12 06:51

    LMK04832的寄存器配置順序嗎?

    大家好,請問該芯片的寄存器配置順序嗎?只要按照TICS軟件中的配置順序就行了嗎?
    發表于 11-11 08:31

    SpringBean初始順序

    Spring bean是Spring框架在運行時管理的對象。Spring bean是任何Spring應用程序的基本構建塊。我們編寫的大多數應用程序邏輯代碼都將放在Spring bean中。 執行
    的頭像 發表于 11-06 16:04 ?167次閱讀
    SpringBean初始<b class='flag-5'>化</b><b class='flag-5'>順序</b>

    低功耗模組短信通:短消息發送實例解析

    今天帶大家學習的是低功耗4G模組發送SMS短消息的解析,以Air780E為實例,讓大家了解更透徹。 SMS(短消息服務,ShortMessageService)功能主要用于在蜂窩網絡中傳輸短消息。 移動,聯通卡不需開通VOLTE可以支持短信功能。
    的頭像 發表于 10-29 14:43 ?1141次閱讀
    低功耗模組短信通:短消息發送<b class='flag-5'>實例</b><b class='flag-5'>解析</b>

    labview實現DBC在界面加載配置

    labview實現DBC在界面加載配置
    發表于 08-19 14:27 ?27次下載

    labview CAN DBC加載解析程序

    labview CAN DBC加載解析程序
    發表于 08-18 11:42

    【電磁兼容標準解析分享】汽車電子部件EMC標準解析---你應該了解和知道的細節(二)

    【電磁兼容標準解析分享】汽車電子部件EMC標準解析---你應該了解和知道的細節(二)
    的頭像 發表于 08-08 08:17 ?4950次閱讀
    【電磁兼容標準<b class='flag-5'>解析</b>分享】汽車電子<b class='flag-5'>零</b>部件EMC標準<b class='flag-5'>解析</b>---你應該了解和知道的細節(二)

    運動控制器的代碼運行順序是什么

    組成部分和關鍵步驟。 初始 運動控制器的代碼運行順序首先從初始化開始。初始是為控制器設置初始狀態的過程,包括配置輸入/輸出接口、設置寄存器、初始
    的頭像 發表于 06-13 09:25 ?470次閱讀

    PLC順序啟動逆順序停止電路如何運行

    了解了順序啟動電路,今天再來看看順序啟動逆順序停止電路如何運行。
    發表于 04-09 14:13 ?762次閱讀
    PLC<b class='flag-5'>順序</b>啟動逆<b class='flag-5'>順序</b>停止電路如何運行

    PLC順序功能圖(SFC)編程深入解析

    稱為順序功能圖 (SFC) 的圖形編程語言由 IEC 61131-3 標準。該圖表從上到下閱讀,塊用于表示步驟、轉換和功能。這種編程風格類似于功能框圖,并且具有一些相同的缺陷,這些缺陷會產生復雜性并降低編程效率。
    發表于 03-21 11:45 ?9898次閱讀
    PLC<b class='flag-5'>順序</b>功能圖(SFC)編程深入<b class='flag-5'>解析</b>

    MCU單片機GPIO初始該按什么順序配置?為什么初始化時有電平跳變?

    GPIO初始化時有時鐘配置、模式配置、輸出配置、復用配置,那么在編寫初始代碼時,到底該按什么順序
    的頭像 發表于 02-22 11:07 ?1528次閱讀
    MCU單片機GPIO初始<b class='flag-5'>化</b>該按什么<b class='flag-5'>順序</b><b class='flag-5'>配置</b>?為什么初始化時有電平跳變?
    主站蜘蛛池模板: 欧美日韩一二区旡码高清在线| 俄罗斯大肥BBXX| 一个人免费完整观看日本| 同时被两个男人轮流舔| 青青草国产精品| 免费观看a视频| 久啪久久全部视频在线| 护士日本xx厕所| 狠狠色狠狠色综合系列| 国产精品日本不卡一区二区| 冈本视频黄页正版| 床伴在线观看免费高清完整泰剧第四集 | 亚洲 在线 日韩 欧美| 歪歪漫画羞羞漫画国产| 天堂so导航| 午夜向日葵高清在线观看| 同桌别揉我奶了嗯啊| 无套内射CHINESEHD| 午夜国产精品视频| 亚洲AV久久婷婷蜜臀无码不卡| 乌克兰10一12x video| 性夜夜春夜夜爽AA片A| 午夜欧洲亚洲AV永久无码精品| 我们中文在线观看免费完整版| 无码11久岁箩筣| 亚洲国产精品嫩草影院永久| 亚洲精品6久久久久中文字幕| 亚洲精品国产专区91在线| 亚洲乱码国产一区三区| 一区两区三不卡| 91看片淫黄大片.在线天堂| a级成人免费毛片完整版| 成人精品在线视频| 国产成人永久免费视频| 国产亚洲精品精华液| 久久99re6热在线播放| 美女乱草鲍高清照片| 欧美伊人久久大香线蕉综合69| 日本免费一区二区三区最新vr | 国产亚洲精品久久久久久白晶晶| 果冻传媒完整免费网站在线观看 |