淺談SpringBoot Bean加載優(yōu)先級(jí)的問題
Bean加載優(yōu)先級(jí)的問題
spring容器載入bean順序是不確定的,spring框架沒有約定特定順序邏輯規(guī)范。但spring保證如果A依賴B(如beanA中有@Autowired B的變量),那么B將先于A被加載。
同一個(gè)類中加載順序
Constructor >> @Autowired >>@ PostConstruct>>@Bean
@DependsOn控制順序
如果A不依賴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è)配置類,注意@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框架沒有約定特定順序邏輯規(guī)范。但spring保證如果A依賴B(如beanA中有@Autowired B的變量),那么B將先于A被加載。但如果beanA不直接依賴B,我們?nèi)绾巫孊仍先加載呢?
控制bean初始化順序
可能有些場(chǎng)景中,bean A 間接依賴 bean B。如Bean B應(yīng)該需要更新一些全局緩存,可能通過單例模式實(shí)現(xiàn)且沒有在spring容器注冊(cè),bean A需要使用該緩存;因此,如果bean B沒有準(zhǔn)備好,bean A無法訪問。
另一個(gè)場(chǎng)景中,bean A是事件發(fā)布者(或JMS發(fā)布者),bean B (或一些) 負(fù)責(zé)監(jiān)聽這些事件,典型的如觀察者模式。我們不想B 錯(cuò)過任何事件,那么B需要首先被初始化。
簡(jiǎn)言之,有很多場(chǎng)景需要bean B應(yīng)該被先于bean A被初始化,從而避免各種負(fù)面影響。我們可以在bean A上使用@DependsOn注解,告訴容器bean B應(yīng)該先被初始化。下面通過示例來說明。
示例說明
示例通過事件機(jī)制說明,Person和Man,然后通過spring配置運(yùn)行。為了方便說明,示例進(jìn)行了簡(jiǎn)化。
Person類
public class Person {
public static void say(){
System.out.println("person.say():Im a person");
}
}
Man類
public class Man {
public void say(){
System.out.println("man.say():Im a man:");
}
}
AppConfig.java
配置運(yùn)行類。
@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)過程中,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)文章
Spring Boot非Web項(xiàng)目運(yùn)行的方法
這篇文章主要給大家介紹了關(guān)于Spring Boot非Web項(xiàng)目運(yùn)行的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用Spring Boot具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧2019-09-09
Java利用ElasticSearch實(shí)現(xiàn)自動(dòng)補(bǔ)全功能
這篇文章主要為大家詳細(xì)介紹了Java如何利用ElasticSearch實(shí)現(xiàn)跟谷歌和百度類似的下拉補(bǔ)全提示功能,文中的示例代碼講解詳細(xì),需要的可以參考一下2023-08-08
詳解Spring Security如何在權(quán)限中使用通配符
小伙伴們知道,在Shiro中,默認(rèn)是支持權(quán)限通配符的。現(xiàn)在給用戶授權(quán)的時(shí)候,可以一個(gè)權(quán)限一個(gè)權(quán)限的配置,也可以直接用通配符。本文將介紹Spring Security如何在權(quán)限中使用通配符,需要的可以參考一下2022-06-06
關(guān)于ArrayList的動(dòng)態(tài)擴(kuò)容機(jī)制解讀
這篇文章主要介紹了關(guān)于ArrayList的動(dòng)態(tài)擴(kuò)容機(jī)制解讀,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-10-10
解決HttpServletResponse和HttpServletRequest取值的2個(gè)坑
這篇文章主要介紹了解決HttpServletResponse和HttpServletRequest取值的2個(gè)坑問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-12-12

