向Spring IOC 容器動態(tài)注冊bean實現(xiàn)方式
本文的大綱

從一個需求談起
這周遇到了這樣一個需求,從第三方的數(shù)據(jù)庫中獲取值,只是一個簡單的分頁查詢,處理這種問題,我一般都是在配置文件中配置數(shù)據(jù)庫的地址等相關(guān)信息,然后在Spring Configuration 注冊數(shù)據(jù)量連接池的bean,然后再將數(shù)據(jù)庫連接池給JdbcTemplate, 但是這種的缺陷是,假設填錯了數(shù)據(jù)庫地址和密碼,或者換了數(shù)據(jù)庫的地址和密碼,在配置文件里面重啟之后,都需要重啟應用。
我想能不能動態(tài)的向Spring IOC容器中注冊和加載bean呢,項目在界面上填寫數(shù)據(jù)庫的地址、用戶名、密碼,存儲之后,將JdbcTemplate和另一個數(shù)據(jù)庫連接池加載到IOC容器中。答案是可以的,我經(jīng)過一番搜索寫出了如下代碼:
@Component
public class BeanDynamicRegister {
private final ConfigurableApplicationContext configurableApplicationContext;
public BeanDynamicRegister(ConfigurableApplicationContext configurableApplicationContext) {
this.configurableApplicationContext = configurableApplicationContext;
}
/**
* 此方法提供出去,供其他bean動態(tài)的向IOC容器中注冊bean。
* 代表使用構(gòu)造器給bean賦值
*
* @param beanName bean名
* @param clazz bean類
* @param args 用于向bean的構(gòu)造函數(shù)中添加值 如果loadType是set,則要求傳遞map.map的key為屬性名,value為屬性值
* @param <T> 返回一個泛型
* @param loadType
* @return
*/
public <T> T registerBeanByLoadType(String beanName, Class<T> clazz, LoadType loadType, Object... args) {
BeanDefinitionBuilder beanDefinitionBuilder = BeanDefinitionBuilder.genericBeanDefinition(clazz);
if (args.length > 0) {
// 將參數(shù)加入到構(gòu)造函數(shù)中
switch (loadType) {
case CONSTRUCTOR:
for (Object arg : args) {
beanDefinitionBuilder.addConstructorArgValue(arg);
}
break;
case SETTER:
Map<String, Object> propertyMap = (Map<String, Object>) args[0];
for (Map.Entry<String, Object> stringObjectEntry : propertyMap.entrySet()) {
beanDefinitionBuilder.addPropertyValue(stringObjectEntry.getKey(), stringObjectEntry.getValue());
}
break;
default:
break;
}
}
BeanDefinition beanDefinition = beanDefinitionBuilder.getRawBeanDefinition();
BeanDefinitionRegistry beanDefinitionRegistry = (BeanDefinitionRegistry) configurableApplicationContext.getBeanFactory();
beanDefinitionRegistry.registerBeanDefinition(beanName, beanDefinition);
return configurableApplicationContext.getBean(beanName, clazz);
}
public <T> T getBeanByName(String beanName,Class<T> requiredType){
return configurableApplicationContext.getBean(beanName,requiredType);
}
/**
* 如果用戶換了地址和密碼,向IOC容器中移除bean。 重新注冊
*
* @param beanName
*/
public void removeBean(String beanName) {
BeanDefinitionRegistry beanDefinitionRegistry = (BeanDefinitionRegistry) configurableApplicationContext.getBeanFactory();
beanDefinitionRegistry.removeBeanDefinition(beanName);
}
}
@SpringBootTest
class SsmApplicationTests {
@Autowired
private LoadBeanService loadBeanService;
private NamedParameterJdbcTemplate jdbcTemplate;
@Autowired
private BeanDynamicRegister beanDynamicRegister;
@Test
public void test() {
loadBeanService.loadDataSourceTest("root", "root");
jdbcTemplate = beanDynamicRegister.getBeanByName("jdbcTemplateOne", NamedParameterJdbcTemplate.class);
System.out.println("--------" + jdbcTemplate);
}
}結(jié)果:

我們就到這里了嗎? 我們觀察一下上面將一個bean加載到Spring IOC容器里經(jīng)過了幾步:
- BeanDefineBuilder 構(gòu)造BeanDefinition
- 然后BeanDefinitionRegistry將其注冊到IOC容器中。(這一步事實上只完成了注冊,還未完成Bean的實例化,屬性填充)
聯(lián)系我們前面的文章《Spring Bean 的生命周期》,我們將Spring 的生命周期理解為“Spring 給我們提供的一些擴展接口,如果bean實現(xiàn)了這些這些接口,應用在啟動的過程中會回調(diào)這些接口的方法。” , 這個理解并不完善,缺少了解析BeanDefinition這個階段。
Spring Bean的生命周期再完善
BeanDefinition
那BeanDefinition是什么? BeanDefinition是一個接口,我們進Spring 官網(wǎng)(https://docs.spring.io/spring...)大致看一下:
A bean definition can contain a lot of configuration information, including constructor arguments, property values, and container-specific information, such as the initialization method, a static factory method name, and so on. A child bean definition inherits configuration data from a parent definition. The child definition can override some values or add others as needed. Using parent and child bean definitions can save a lot of typing. Effectively, this is a form of templating.
bean 的定義信息可以包含許多配置信息,包括構(gòu)造函數(shù)參數(shù),屬性值和特定于容器的信息,例如初始化方法,靜態(tài)工廠方法名稱等。子 bean 定義可以從父 bean 定義繼承配置數(shù)據(jù)。子 bean 的定義信息可以覆蓋某些值,或者可以根據(jù)需要添加其他值。使用父 bean 和子 bean 的定義可以節(jié)省很多輸入(實際上,這是一種模板的設計形式)。
這段說的可能有點抽象, 你點BeanDefinition進去,你就會發(fā)現(xiàn)有很多熟悉的面孔:

Bean的作用域: 單例,還是多例。

lazyInit是否是懶加載。
這些都是描述Spring Bean的信息,我們可以類比到Java中的類,每個類都會有class屬性,我們在配置類或者xml中的配置Bean的元信息,也被映射到這里。供IOC容器將Bean加入時使用。所以我們可以為對Spring Bean的生命周期的理解打一個補丁:
- 從xml或配置類中解析BeanDefintion
- BeanDefinition 注冊,此時還未完成Bean的實例化。
我們可以打斷點來驗證一下:


- Bean 實例化
- Bean的屬性賦值+依賴注入
- Bean的初始化階段的方法回調(diào)
- Bean的銷毀。

Bean 加入IOC容器的幾種方式
我們這里再來總結(jié)一下一個Bean注入Spring IOC容器的幾種形式:
啟動時加入
- 配置類: @Configuration+@Bean
- 配置文件: xml
注解形式
- @Component
- @Service
- @Controller
- @Repository
- @import
- @Qualifier
- @Resource
- @Inject
運行時加入
這三種最終都是通過BeanDefinitionRegistry來注入的,ImportBeanDefinitionRegistrar是一個接口,留給我們實現(xiàn)的方法如下:
default void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
}- ImportBeanDefinitionRegistrar
- 手動構(gòu)造BeanDefinition注入(我們上面就是自己手動構(gòu)造BeanDefinition注入)
- 借助BeanDefinitionRegistryPostProcessor注入
? BeanDefinitionRegistryPostProcessor也是一個接口,留給我們實現(xiàn)的方法如下:
void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException;
從spring容器中動態(tài)添加或移除bean
public class DemoUtil {
@Autowired
private ApplicationContext applicationContext;
//添加bean
public void addBean(String beanName, Class<?> beanClass) {
BeanDefinitionRegistry beanDefinitionRegistry = (BeanDefinitionRegistry) applicationContext.getAutowireCapableBeanFactory();
BeanDefinitionBuilder beanDefinitionBuilder = BeanDefinitionBuilder.genericBeanDefinition(beanClass);
BeanDefinition beanDefinition = beanDefinitionBuilder.getBeanDefinition();
if (!beanDefinitionRegistry.containsBeanDefinition(beanName)) {
beanDefinitionRegistry.registerBeanDefinition(beanName, beanDefinition);
}
}
//移除bean
public void removeBean(String beanName) {
BeanDefinitionRegistry beanDefinitionRegistry = (BeanDefinitionRegistry) applicationContext.getAutowireCapableBeanFactory();
beanDefinitionRegistry.getBeanDefinition(beanName);
beanDefinitionRegistry.removeBeanDefinition(beanName);
}
}
參考資料
- 180804-Spring之動態(tài)注冊bean http://www.dhdzp.com/article/145136.htm
- 從spring容器中動態(tài)添加或移除bean
- 《從 0 開始深入學習 Spring》http://www.dhdzp.com/books/478522.html
以上就是向Spring IOC 容器動態(tài)注冊bean實現(xiàn)方式的詳細內(nèi)容,更多關(guān)于Spring IOC 容器動態(tài)注冊bean的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Java的MyBatis框架中實現(xiàn)多表連接查詢和查詢結(jié)果分頁
這篇文章主要介紹了Java的MyBatis框架中實現(xiàn)多表連接查詢和查詢結(jié)果分頁,借助MyBatis框架中帶有的動態(tài)SQL查詢功能可以比普通SQL查詢做到更多,需要的朋友可以參考下2016-04-04
Spring Boot + Kotlin整合MyBatis的方法教程
前幾天由于工作需要,便開始學習了kotlin,java基礎(chǔ)扎實學起來也還算比較快,對于kotlin這個編程語言自然是比java有趣一些,下面這篇文章主要給大家介紹了關(guān)于Spring Boot + Kotlin整合MyBatis的方法教程,需要的朋友可以參考下。2018-01-01
Java 利用枚舉實現(xiàn)接口進行統(tǒng)一管理
這篇文章主要介紹了Java 利用枚舉實現(xiàn)接口進行統(tǒng)一管理,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-02-02
springboot(thymeleaf)中th:field和th:value的區(qū)別及說明
這篇文章主要介紹了springboot(thymeleaf)中th:field和th:value的區(qū)別及說明,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-10-10
JAVA裝飾者模式(從現(xiàn)實生活角度理解代碼原理)
裝飾者模式可以動態(tài)地給一個對象添加一些額外的職責。就增加功能來說,Decorator模式相比生成子類更為靈活。這篇文章主要介紹了JAVA裝飾者模式的相關(guān)資料,需要的朋友可以參考下2016-12-12

