淺談SpringBoot Bean加載優(yōu)先級(jí)的問(wèn)題
Bean加載優(yōu)先級(jí)的問(wèn)題
spring容器載入bean順序是不確定的,spring框架沒(méi)有約定特定順序邏輯規(guī)范。但spring保證如果A依賴(lài)B(如beanA中有@Autowired B的變量),那么B將先于A被加載。
同一個(gè)類(lèi)中加載順序
Constructor >> @Autowired >>@ PostConstruct>>@Bean
@DependsOn控制順序
如果A不依賴(lài)B,但是A需要在B后面初始化,可以使用@DependsOn(value=“Bbeanname”)。B的@Bean上面需要手動(dòng)指定Name,否則找不到。
@Order不能控制順序
@Order注解并不能改變Bean加載優(yōu)先級(jí),@Order注解用于設(shè)置裝載到list中Bean的順序
@Order(2) @Component public class AnoBean1 implements IBean { private String name = "ano order bean 1"; public AnoBean1() { System.out.println(name); } } @Order(1) @Component public class AnoBean2 implements IBean { private String name = "ano order bean 2"; public AnoBean2() { System.out.println(name); } } @Component public class AnoTestBean { public AnoTestBean(List<IBean> anoBeanList) { for (IBean bean : anoBeanList) { System.out.println("in ano testBean: " + bean.getClass().getName()); } } }
上面代碼輸出結(jié)果
ano order bean 1
ano order bean 2
in ano testBean: AnoBean2
in ano testBean: AnoBean1
Spring控制Bean加載順序
使用Spring @Order控制bean加載順序
兩個(gè)演示bean
package com.ziyear.spring4_2.order; public class Demo1Service { }
package com.ziyear.spring4_2.order; public class Demo2Service { }
兩個(gè)配置類(lèi),注意@Order配置加載的順序
package com.ziyear.spring4_2.order; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.core.annotation.Order; @Configuration @Order(2) public class Demo1Config { @Bean public Demo1Service demo1Service(){ System.out.println("demo1config 加載了"); return new Demo1Service(); } }
package com.ziyear.spring4_2.order; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.core.annotation.Order; @Configuration @Order(1) public class Demo2Config { @Bean public Demo2Service demo2Service(){ System.out.println("demo2config 加載了"); return new Demo2Service(); } }
運(yùn)行
package com.ziyear.spring4_2.order; import org.springframework.context.annotation.AnnotationConfigApplicationContext; public class Main { public static void main(String[] args) { AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext("com.ziyear.spring4_2.order"); } }
輸出結(jié)果
demo2config 加載了
demo1config 加載了
使用Spring @DependsOn控制bean加載順序
spring容器載入bean順序是不確定的,spring框架沒(méi)有約定特定順序邏輯規(guī)范。但spring保證如果A依賴(lài)B(如beanA中有@Autowired B的變量),那么B將先于A被加載。但如果beanA不直接依賴(lài)B,我們?nèi)绾巫孊仍先加載呢?
控制bean初始化順序
可能有些場(chǎng)景中,bean A 間接依賴(lài) bean B。如Bean B應(yīng)該需要更新一些全局緩存,可能通過(guò)單例模式實(shí)現(xiàn)且沒(méi)有在spring容器注冊(cè),bean A需要使用該緩存;因此,如果bean B沒(méi)有準(zhǔn)備好,bean A無(wú)法訪(fǎng)問(wèn)。
另一個(gè)場(chǎng)景中,bean A是事件發(fā)布者(或JMS發(fā)布者),bean B (或一些) 負(fù)責(zé)監(jiān)聽(tīng)這些事件,典型的如觀察者模式。我們不想B 錯(cuò)過(guò)任何事件,那么B需要首先被初始化。
簡(jiǎn)言之,有很多場(chǎng)景需要bean B應(yīng)該被先于bean A被初始化,從而避免各種負(fù)面影響。我們可以在bean A上使用@DependsOn注解,告訴容器bean B應(yīng)該先被初始化。下面通過(guò)示例來(lái)說(shuō)明。
示例說(shuō)明
示例通過(guò)事件機(jī)制說(shuō)明,Person和Man,然后通過(guò)spring配置運(yùn)行。為了方便說(shuō)明,示例進(jìn)行了簡(jiǎn)化。
Person類(lèi)
public class Person { public static void say(){ System.out.println("person.say():Im a person"); } }
Man類(lèi)
public class Man { public void say(){ System.out.println("man.say():Im a man:"); } }
AppConfig.java
配置運(yùn)行類(lèi)。
@Configuration @ComponentScan("com.ziyear.demo") public class Appconfig { @Bean(initMethod = "say") @DependsOn("man") public Person personBean () { return new Person(); } @Bean(name = "man", initMethod = "say") //@Lazy public Man manBean () { return new Man(); } public static void main (String[] strings) { new AnnotationConfigApplicationContext(Appconfig.class); } }
運(yùn)行AppConfig的main方法,輸出結(jié)果為:
man.say():Im a man:
person.say():Im a person
小結(jié)一下
如果我們注釋掉@DependsOn(“man”),我們可能不確定獲得相同結(jié)果。嘗試多次運(yùn)行main方法,偶爾會(huì)看到不同的say方法別執(zhí)行。為什么是偶爾呢?因?yàn)槿萜鲉?dòng)過(guò)程中,spring按任意順序加載bean。
那么當(dāng)不使用@DependsOn可以讓其100%確定嗎?可以使用@Lazy注解放在manBean ()上。因?yàn)镸an在啟動(dòng)階段不加載,當(dāng)其他bean需要其時(shí)才加載。這次我們僅Person被初始化。
person.say():Im a person
現(xiàn)在從新增加@DependsOn,也不刪除@Lazy注解,輸出結(jié)果和第一次一致,雖然我們使用了@Lazy注解,Man在啟動(dòng)時(shí)仍然被加載,因?yàn)锧DependsOn表明需要Man。
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
java配置dbcp連接池(數(shù)據(jù)庫(kù)連接池)示例分享
java配置dbcp連接池示例分享,大家參考使用吧2013-12-12SpringBoot使用Jasypt對(duì)配置文件和數(shù)據(jù)庫(kù)密碼加密
在做數(shù)據(jù)庫(kù)敏感信息保護(hù)時(shí),應(yīng)加密存儲(chǔ),本文就來(lái)介紹一下SpringBoot使用Jasypt對(duì)配置文件和數(shù)據(jù)庫(kù)密碼加密,具有一定的參考價(jià)值,感興趣的可以了解一下2024-02-02java中處理stream.filter()的實(shí)例代碼
stream()是Java 8中的一個(gè)函數(shù)式接口,用于處理數(shù)據(jù)流,它可以從一個(gè)數(shù)據(jù)源,如集合,數(shù)組等生成一個(gè)流,這篇文章主要給大家介紹了關(guān)于java中處理stream.filter()的相關(guān)資料,需要的朋友可以參考下2024-08-08SpringBoot+Apache tika實(shí)現(xiàn)文檔內(nèi)容解析的示例詳解
Apache tika是Apache開(kāi)源的一個(gè)文檔解析工具,本文主要為大家介紹了如何在springboot中引入tika的方式解析文檔,感興趣的小伙伴可以了解一下2023-07-07詳解使用spring aop實(shí)現(xiàn)業(yè)務(wù)層mysql 讀寫(xiě)分離
本篇文章主要介紹了使用spring aop實(shí)現(xiàn)業(yè)務(wù)層mysql 讀寫(xiě)分離,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-01-01java String、StringBuilder和StringBuffer的區(qū)別詳解
這篇文章主要介紹了java String、StringBuilder和StringBuffer的區(qū)別詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-01-01java 中動(dòng)態(tài)代理詳解及實(shí)例
這篇文章主要介紹了java 中動(dòng)態(tài)代理詳解及實(shí)例的相關(guān)資料,需要的朋友可以參考下2017-06-06