Spring控制Bean加載順序的操作方法
前言
正常情況下,Spring 容器加載 Bean 的順序是不確定的,那么我們?nèi)绻枰错樞蚣虞d Bean 時應(yīng)如何操作?本文將詳細(xì)講述我們?nèi)绾尾拍芸刂?Bean 的加載順序。
場景
我創(chuàng)建了 4 個 Class 文件,分別命名為
- FirstInitialization
- SecondInitialization
- ThirdInitialization
- ForthInitialization
我希望這 4 個類按照 1、2、3、4 的順序加載。
如下圖,直接加載的話,順序是 1、4、2、3,并不能達(dá)到要求。

如何控制
注意:網(wǎng)上很多文章說
Order注解或Ordered接口可以控制 Bean 的加載順序,其是并不能,它們的作用是定義 Spring IOC 容器中 Bean 定義類的執(zhí)行順序的優(yōu)先級,并不是定義加載順序。
使用@DependsOn 注解
在需要調(diào)整順序的類上依次加@DependsOn注解,缺點是類過多的時候需要一個個加注解,且不好維護(hù)。
@Component
public class FirstInitialization {
@PostConstruct
public void init(){
System.out.println("我是第一個加載!");
}
}
@Component
@DependsOn("firstInitialization")
public class SecondInitialization {
@PostConstruct
public void init(){
System.out.println("我是第二個加載!");
}
}
@Component
@DependsOn("secondInitialization")
public class ThirdInitialization {
@PostConstruct
public void init(){
System.out.println("我是第三個加載!");
}
}
@Component
@DependsOn("thirdInitialization")
public class ForthInitialization {
@PostConstruct
public void init(){
System.out.println("我是第四個加載!");
}
}
執(zhí)行結(jié)果如下

基于 ApplicationContextInitializer 接口
接口簡介
這里我簡單介紹一個這個接口的用處, 等到整理到相關(guān)源碼的時候再詳細(xì)介紹。
ApplicationContextInitializer接口是在 Spring 容器刷新之前執(zhí)行的一個回調(diào)函數(shù)。
執(zhí)行時機(jī):
Spring 內(nèi)部執(zhí)行
ConfigurableApplicationContext#refresh()方法前;SpringBoot 執(zhí)行
run()方法前。
一般有什么用呢?
在 SpringBoot 應(yīng)用中 Classpath 上會有很多 jar 包,有些 jar 包需要在refresh()調(diào)用前對應(yīng)用上下文做一些初始化動作,因此會提供ApplicationContextInitializer接口的實現(xiàn)類,放在如下圖的文件中,這樣會被SpringApplication#initialize發(fā)現(xiàn),然后完成對應(yīng)初始化。

實現(xiàn)步驟
首先創(chuàng)建一個類繼承ApplicationContextInitializer接口。
public class MyApplicationContextInitializer implements ApplicationContextInitializer {
@Override
public void initialize(ConfigurableApplicationContext applicationContext) {
//將自定義的BeanFactoryPostProcessor實現(xiàn)類保存到ApplicationContext中
applicationContext.addBeanFactoryPostProcessor(new MyBeanFactoryPostProcessor());
}
}
創(chuàng)建META-INF/spring.factories文件。

自定義BeanDefinitionRegistryPostProcessor。
/**
* BeanFactoryPostProcessor的子類
* 允許開發(fā)人員在Bean定義注冊之前和之后對BeanDefinition進(jìn)行自定義處理,例如添加,修改或刪除Bean定義等。
*/
public class MyBeanFactoryPostProcessor implements BeanDefinitionRegistryPostProcessor {
// 初始化需要排序的類,這里要保證插入順序只能用LinkedHashMap
private static final Map<String, Class> ORDER_BEAN_MAP = new LinkedHashMap<>() {
{
put("firstInitialization", FirstInitialization.class);
put("secondInitialization", SecondInitialization.class);
put("thirdInitialization", ThirdInitialization.class);
put("forthInitialization", ForthInitialization.class);
}
};
@Override
public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException {
Optional.ofNullable(ORDER_BEAN_MAP.keySet()).orElse(new HashSet<>()).stream()
.forEach(beanName -> {
// 初始化一個 Bean 定義
AbstractBeanDefinition beanDefinition = BeanDefinitionBuilder
.genericBeanDefinition().getBeanDefinition();
// 按順序注冊每個Bean
beanDefinition.setBeanClass(ORDER_BEAN_MAP.get(beanName));
registry.registerBeanDefinition(beanName, beanDefinition);
});
}
}
執(zhí)行結(jié)果如下

以上就是Spring控制Bean加載順序的操作方法的詳細(xì)內(nèi)容,更多關(guān)于Spring控制Bean加載順序的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
servlet監(jiān)聽實現(xiàn)統(tǒng)計在線人數(shù)功能 附源碼下載
這篇文章主要為大家詳細(xì)介紹了servlet監(jiān)聽統(tǒng)計在線人數(shù)的實現(xiàn)方法,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-04-04
關(guān)于Spring Boot和Kotlin的聯(lián)合開發(fā)
這篇文章主要介紹了關(guān)于Spring Boot和Kotlin的聯(lián)合開發(fā),需要的朋友可以參考下2017-06-06
Springboot 整合通用mapper和pagehelper展示分頁數(shù)據(jù)的問題(附github源碼)
這篇文章主要介紹了Springboot 整合通用mapper和pagehelper展示分頁數(shù)據(jù)(附github源碼),本文通過實例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-09-09

