Spring注解驅(qū)動(dòng)擴(kuò)展原理BeanFactoryPostProcessor
1、擴(kuò)展原理-BeanFactoryPostProcessor
BeanFactoryPostProcessor
* 擴(kuò)展原理:
* BeanPostProcessor:bean后置處理器,bean創(chuàng)建對(duì)象初始化前后進(jìn)行攔截工作的
*
* 1、BeanFactoryPostProcessor:beanFactory的后置處理器;
* 在BeanFactory標(biāo)準(zhǔn)初始化之后調(diào)用,來定制和修改BeanFactory的內(nèi)容;
* 所有的bean定義已經(jīng)保存加載到beanFactory,但是bean的實(shí)例還未創(chuàng)建
*
*
* BeanFactoryPostProcessor原理:
* 1)、ioc容器創(chuàng)建對(duì)象
* 2)、invokeBeanFactoryPostProcessors(beanFactory);
* 如何找到所有的BeanFactoryPostProcessor并執(zhí)行他們的方法;
* 1)、直接在BeanFactory中找到所有類型是BeanFactoryPostProcessor的組件,并執(zhí)行他們的方法
* 2)、在初始化創(chuàng)建其他組件前面執(zhí)行
代碼實(shí)現(xiàn)
@Component public class MyBeanFactoryPostProcessor implements BeanFactoryPostProcessor { public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException { System.out.println("MyBeanFactoryPostProcessor...postProcessBeanFactory..."); int count = beanFactory.getBeanDefinitionCount(); String[] names = beanFactory.getBeanDefinitionNames(); System.out.println("當(dāng)前BeanFactory中有"+count+" 個(gè)Bean"); System.out.println(Arrays.asList(names)); } }
2、擴(kuò)展原理-BeanDefinitionRegistryPostProcessor
BeanDefinitionRegistryPostProcessor
* 2、BeanDefinitionRegistryPostProcessor extends BeanFactoryPostProcessor
* postProcessBeanDefinitionRegistry();
* 在所有bean定義信息將要被加載,bean實(shí)例還未創(chuàng)建的;
*
* 優(yōu)先于BeanFactoryPostProcessor執(zhí)行;
* 利用BeanDefinitionRegistryPostProcessor給容器中再額外添加一些組件;
*
* 原理:
* 1)、ioc創(chuàng)建對(duì)象
* 2)、refresh()-》invokeBeanFactoryPostProcessors(beanFactory);
* 3)、從容器中獲取到所有的BeanDefinitionRegistryPostProcessor組件。
* 1、依次觸發(fā)所有的postProcessBeanDefinitionRegistry()方法
* 2、再來觸發(fā)postProcessBeanFactory()方法BeanFactoryPostProcessor;
*
* 4)、再來從容器中找到BeanFactoryPostProcessor組件;然后依次觸發(fā)postProcessBeanFactory()方法
代碼實(shí)現(xiàn)
@Component public class MyBeanDefinitionRegistryPostProcessor implements BeanDefinitionRegistryPostProcessor{ public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException { // TODO Auto-generated method stub System.out.println("MyBeanDefinitionRegistryPostProcessor...bean的數(shù)量:"+beanFactory.getBeanDefinitionCount()); } //BeanDefinitionRegistry Bean定義信息的保存中心,以后BeanFactory就是按照BeanDefinitionRegistry里面保存的每一個(gè)bean定義信息創(chuàng)建bean實(shí)例; public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException { // TODO Auto-generated method stub System.out.println("postProcessBeanDefinitionRegistry...bean的數(shù)量:"+registry.getBeanDefinitionCount()); //RootBeanDefinition beanDefinition = new RootBeanDefinition(Blue.class); AbstractBeanDefinition beanDefinition = BeanDefinitionBuilder.rootBeanDefinition(Blue.class).getBeanDefinition(); registry.registerBeanDefinition("hello", beanDefinition); } }
3、擴(kuò)展原理-ApplicationListener用法
監(jiān)聽器ApplicationListener
* 3、ApplicationListener:監(jiān)聽容器中發(fā)布的事件。事件驅(qū)動(dòng)模型開發(fā);
* public interface ApplicationListener<E extends ApplicationEvent>
* 監(jiān)聽 ApplicationEvent 及其下面的子事件;
*
* 步驟:
* 1)、寫一個(gè)監(jiān)聽器(ApplicationListener實(shí)現(xiàn)類)來監(jiān)聽某個(gè)事件(ApplicationEvent及其子類)
* @EventListener;
* 原理:使用EventListenerMethodProcessor處理器來解析方法上的@EventListener;
*
* 2)、把監(jiān)聽器加入到容器;
* 3)、只要容器中有相關(guān)事件的發(fā)布,我們就能監(jiān)聽到這個(gè)事件;
* ContextRefreshedEvent:容器刷新完成(所有bean都完全創(chuàng)建)會(huì)發(fā)布這個(gè)事件;
* ContextClosedEvent:關(guān)閉容器會(huì)發(fā)布這個(gè)事件;
* 4)、發(fā)布一個(gè)事件:
* applicationContext.publishEvent();
代碼實(shí)現(xiàn):
方式一:實(shí)現(xiàn)ApplicationListener<E extends ApplicationEvent>接口
@Component public class MyApplicationListener implements ApplicationListener<ApplicationEvent> { //當(dāng)容器中發(fā)布此事件以后,方法觸發(fā) public void onApplicationEvent(ApplicationEvent event) { // TODO Auto-generated method stub System.out.println("收到事件:"+event); } }
方式二:使用@EventListener注解標(biāo)識(shí)事件監(jiān)聽方法
@Service public class UserService { @EventListener(classes={ApplicationEvent.class}) public void listen(ApplicationEvent event){ System.out.println("UserService。。監(jiān)聽到的事件:"+event); } }
4、擴(kuò)展原理-ApplicationListener原理
* 原理:
* ContextRefreshedEvent、IOCTest_Ext$1[source=我發(fā)布的時(shí)間]、ContextClosedEvent;
* 1)、ContextRefreshedEvent事件:
* 1)、容器創(chuàng)建對(duì)象:refresh();
* 2)、finishRefresh();容器刷新完成會(huì)發(fā)布ContextRefreshedEvent事件
* 2)、自己發(fā)布事件;
* 3)、容器關(guān)閉會(huì)發(fā)布ContextClosedEvent;
*
* 【事件發(fā)布流程】:
* 3)、publishEvent(new ContextRefreshedEvent(this));
* 1)、獲取事件的多播器(派發(fā)器):getApplicationEventMulticaster()
* 2)、multicastEvent派發(fā)事件:
* 3)、獲取到所有的ApplicationListener;
* for (final ApplicationListener<?> listener : getApplicationListeners(event, type)) {
* 1)、如果有Executor,可以支持使用Executor進(jìn)行異步派發(fā);
* Executor executor = getTaskExecutor();
* 2)、否則,同步的方式直接執(zhí)行l(wèi)istener方法;invokeListener(listener, event);
* 拿到listener回調(diào)onApplicationEvent方法;
*
* 【事件多播器(派發(fā)器)】
* 1)、容器創(chuàng)建對(duì)象:refresh();
* 2)、initApplicationEventMulticaster();初始化ApplicationEventMulticaster;
* 1)、先去容器中找有沒有id=“applicationEventMulticaster”的組件;
* 2)、如果沒有this.applicationEventMulticaster = new SimpleApplicationEventMulticaster(beanFactory);
* 并且加入到容器中,我們就可以在其他組件要派發(fā)事件,自動(dòng)注入這個(gè)applicationEventMulticaster;
*
* 【容器中有哪些監(jiān)聽器】
* 1)、容器創(chuàng)建對(duì)象:refresh();
* 2)、注冊(cè)監(jiān)聽器:registerListeners();
* 從容器中拿到所有的監(jiān)聽器,把他們注冊(cè)到applicationEventMulticaster中;
* String[] listenerBeanNames = getBeanNamesForType(ApplicationListener.class, true, false);
* //將listener注冊(cè)到ApplicationEventMulticaster中
* getApplicationEventMulticaster().addApplicationListenerBean(listenerBeanName);
5、擴(kuò)展原理-@EventListener與SmartInitializingSingleton
* SmartInitializingSingleton 原理:->afterSingletonsInstantiated();
* 1)、ioc容器創(chuàng)建對(duì)象并refresh();
* 2)、finishBeanFactoryInitialization(beanFactory);初始化剩下的單實(shí)例bean;
* 1)、先創(chuàng)建所有的單實(shí)例bean;getBean();
* 2)、獲取所有創(chuàng)建好的單實(shí)例bean,判斷是否是SmartInitializingSingleton類型的;
* 如果是就調(diào)用afterSingletonsInstantiated();
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Dwr3.0純注解(純Java Code配置)配置與應(yīng)用淺析二之前端調(diào)用后端
我們講到了后端純Java Code的Dwr3配置,完全去掉了dwr.xml配置文件,但是對(duì)于使用注解的類卻沒有使用包掃描,而是在Servlet初始化參數(shù)的classes里面加入了我們的Service組件的聲明暴露,對(duì)于這個(gè)問題需要后面我們?cè)偌?xì)細(xì)研究下這篇文章,主要分析介紹前端怎么直接調(diào)用后端2016-04-04Java實(shí)現(xiàn)Token工具類進(jìn)行登錄和攔截
在應(yīng)用的登錄時(shí)需要生成token進(jìn)行驗(yàn)證,并放入信息,之后的話可以直接使用瀏覽器的session進(jìn)行登錄,本文就來利用java編寫一個(gè)token工具類,可以很方便的生成和解析token,感興趣的可以了解下2023-12-12spring事務(wù)之事務(wù)掛起和事務(wù)恢復(fù)源碼解讀
這篇文章主要介紹了spring事務(wù)之事務(wù)掛起和事務(wù)恢復(fù)源碼解讀,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-11-11SpringSecurity實(shí)現(xiàn)自定義登錄方式
本文介紹自定義登錄流程,包括自定義AuthenticationToken、AuthenticationFilter、AuthenticationProvider以及SecurityConfig配置類,詳細(xì)解析了認(rèn)證流程的實(shí)現(xiàn),為開發(fā)人員提供了具體的實(shí)施指導(dǎo)和參考2024-09-09從dubbo zookeeper注冊(cè)地址提取出zookeeper地址的方法
今天小編就為大家分享一篇關(guān)于從dubbo zookeeper注冊(cè)地址提取出zookeeper地址的方法,小編覺得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來看看吧2018-12-12Java 常見的限流算法詳細(xì)分析并實(shí)現(xiàn)
大數(shù)據(jù)量高并發(fā)訪問時(shí),經(jīng)常出現(xiàn)服務(wù)或接口面對(duì)暴漲的請(qǐng)求而不可用的情況,甚至引發(fā)連鎖反映導(dǎo)致整個(gè)系統(tǒng)崩潰。此時(shí)你需要使用的技術(shù)手段之一就是限流,當(dāng)請(qǐng)求達(dá)到一定的并發(fā)數(shù)或速率,就進(jìn)行等待、排隊(duì)、降級(jí)、拒絕服務(wù)等。限流時(shí),常見算法是計(jì)數(shù)器、漏斗、令牌桶算法2022-04-04解決springboot3.2集成mybatis-plus3.5.4.1報(bào)錯(cuò)的問題
這篇文章給大家介紹了如何解決springboot3.2集成mybatis-plus3.5.4.1報(bào)錯(cuò)的問題,文中通過圖文介紹的非常詳細(xì),具有一定的參考價(jià)值,需要的朋友可以參考下2023-12-12