Spring注解驅(qū)動(dòng)之BeanDefinitionRegistryPostProcessor原理解析
BeanDefinitionRegistryPostProcessor概述
可以看到BeanDefinitionRegistryPostProcessor是BeanFactoryPostProcessor的子接口。
注釋中說(shuō)執(zhí)行時(shí)機(jī)是所有合法的bean定義已經(jīng)加載,但是還沒(méi)實(shí)例化。
看起來(lái)和BeanFactoryPostProcessor執(zhí)行時(shí)機(jī)差不多,但是BeanFactoryPostProcessor的注釋是所有bean定義被加載,而BeanDefinitionRegistryPostProcessor是所有合法的bean定義。
接著看注釋:
This allows for adding further bean definitions before the next post-processing phase kicks in.
意思是BeanDefinitionRegistryPostProcessor允許添加將來(lái)的bean定義在下一個(gè)后置處理器階段開(kāi)始之前。簡(jiǎn)單說(shuō)就是還可以往容器中增加新的bean的定義。
因此,大概率BeanDefinitionRegistryPostProcessor的執(zhí)行順序在BeanFactoryPostProcessor之前。
案例實(shí)踐
首先,編寫一個(gè)類,例如MyBeanDefinitionRegistryPostProcessor,它應(yīng)要實(shí)現(xiàn)BeanDefinitionRegistryPostProcessor這個(gè)接口。
package com.meimeixia.ext; import org.springframework.beans.BeansException; import org.springframework.beans.factory.config.ConfigurableListableBeanFactory; import org.springframework.beans.factory.support.AbstractBeanDefinition; import org.springframework.beans.factory.support.BeanDefinitionBuilder; import org.springframework.beans.factory.support.BeanDefinitionRegistry; import org.springframework.beans.factory.support.BeanDefinitionRegistryPostProcessor; import org.springframework.beans.factory.support.RootBeanDefinition; import org.springframework.stereotype.Component; import com.meimeixia.bean.Blue; // 記住,我們這個(gè)組件寫完之后,一定別忘了給它加在容器中 @Component public class MyBeanDefinitionRegistryPostProcessor implements BeanDefinitionRegistryPostProcessor { @Override public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException { // TODO Auto-generated method stub System.out.println("MyBeanDefinitionRegistryPostProcessor...bean的數(shù)量:" + beanFactory.getBeanDefinitionCount()); } /** * 這個(gè)BeanDefinitionRegistry就是Bean定義信息的保存中心,這個(gè)注冊(cè)中心里面存儲(chǔ)了所有的bean定義信息, * 以后,BeanFactory就是按照BeanDefinitionRegistry里面保存的每一個(gè)bean定義信息來(lái)創(chuàng)建bean實(shí)例的。 * * bean定義信息包括有哪些呢?有這些,這個(gè)bean是單例的還是多例的、bean的類型是什么以及bean的id是什么。 * 也就是說(shuō),這些信息都是存在BeanDefinitionRegistry里面的。 */ @Override public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException { // TODO Auto-generated method stub System.out.println("postProcessBeanDefinitionRegistry...bean的數(shù)量:" + registry.getBeanDefinitionCount()); // 除了查看bean的數(shù)量之外,我們還可以給容器里面注冊(cè)一些bean,我們以前也簡(jiǎn)單地用過(guò) /* * 第一個(gè)參數(shù):我們將要給容器中注冊(cè)的bean的名字 * 第二個(gè)參數(shù):BeanDefinition對(duì)象 */ // RootBeanDefinition beanDefinition = new RootBeanDefinition(Blue.class); // 現(xiàn)在我準(zhǔn)備給容器中添加一個(gè)Blue對(duì)象 // 咱們也可以用另外一種辦法,即使用BeanDefinitionBuilder這個(gè)構(gòu)建器生成一個(gè)BeanDefinition對(duì)象,很顯然,這兩種方法的效果都是一樣的 AbstractBeanDefinition beanDefinition = BeanDefinitionBuilder.rootBeanDefinition(Blue.class).getBeanDefinition(); registry.registerBeanDefinition("hello", beanDefinition); } }
測(cè)試結(jié)果
可以看到,BeanDefinitionRegistryPostProcessor里面的兩個(gè)方法,postProcessBeanDefinitionRegistry在postProcessBeanFactory之前執(zhí)行。
BeanDefinitionRegistryPostProcessor比BeanFactoryPostProcessor先執(zhí)行。
源碼分析
自己在測(cè)試示例中方法打斷點(diǎn),然后查看調(diào)用棧即可,下面是一些主要的代碼片段。
繼續(xù)向下看,可以看到會(huì)取出所有實(shí)現(xiàn)了BeanDefinitionRegistryPostProcessor接口的類,即從容器中獲取到所有的BeanDefinitionRegistryPostProcessor組件。
然后,優(yōu)先調(diào)用實(shí)現(xiàn)了PriorityOrdered接口的BeanDefinitionRegistryPostProcessor組件。
點(diǎn)進(jìn)去這個(gè)方法里面一看究竟,原來(lái)是先調(diào)用完BeanDefinitionRegistryPostProcessor組件里面的postProcessBeanDefinitionRegistry方法,然后再來(lái)調(diào)用它里面的postProcessBeanFactory方法。
我們?cè)賮?lái)仔細(xì)看一下PostProcessorRegistrationDelegate類中的invokeBeanFactoryPostProcessors方法,只不過(guò)這時(shí)是從程序停留的地方(即第122行代碼處)往下看,如下圖所示。
小結(jié)
BeanDefinitionRegistryPostProcessor的執(zhí)行流程。
1. 創(chuàng)建IOC容器。
2. 調(diào)用refresh方法。
3. 從IOC容器中獲取所有的BeanDefinitionRegistryPostProcessor組件,并依次觸發(fā)它們的postProcessBeanDefinitionRegistry方法,之后觸發(fā)它的postProcessBeanFactory方法。
4. 從IOC容器中獲取到所有的BeanFactoryPostProcessor組件,并依次觸發(fā)它們的postProcessBeanFactory方法。
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
測(cè)量Java對(duì)象所占內(nèi)存大小方式
這篇文章主要介紹了測(cè)量Java對(duì)象所占內(nèi)存大小方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-09-09面向切面的Spring通過(guò)切點(diǎn)來(lái)選擇連接點(diǎn)實(shí)例詳解
這篇文章主要為大家介紹了面向切面的Spring通過(guò)切點(diǎn)來(lái)選擇連接點(diǎn)實(shí)例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-10-10springboot如何讀取配置文件(application.yml)中的屬性值
本篇文章主要介紹了springboot如何讀取配置文件(application.yml)中的屬性值,具有一定的參考價(jià)值,有興趣的小伙伴可以了解一下2017-04-04Spring Boot集成Spring Cache過(guò)程詳解
這篇文章主要介紹了Spring Boot集成Spring Cache過(guò)程詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-10-10Java實(shí)現(xiàn)掃雷游戲詳細(xì)代碼講解
windows自帶的游戲《掃雷》是陪伴了無(wú)數(shù)人的經(jīng)典游戲,本文將利用Java語(yǔ)言實(shí)現(xiàn)這一經(jīng)典的游戲,文中的示例代碼講解詳細(xì),感興趣的可以學(xué)習(xí)一下2022-05-05json解析時(shí)遇到英文雙引號(hào)報(bào)錯(cuò)的解決方法
下面小編就為大家分享一篇json解析時(shí)遇到英文雙引號(hào)報(bào)錯(cuò)的解決方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-02-02Java線程讓步y(tǒng)ield用法實(shí)例分析
這篇文章主要介紹了Java線程讓步y(tǒng)ield用法,結(jié)合實(shí)例形式分析了java中yield()方法的功能、原理及線程讓步操作的相關(guān)實(shí)現(xiàn)技巧,需要的朋友可以參考下2019-09-09