您的位置:首页技术文章
文章详情页

Spring注解驱动扩展原理BeanFactoryPostProcessor

【字号: 日期:2023-09-14 10:23:39浏览:6作者:猪猪

1、扩展原理-BeanFactoryPostProcessor

BeanFactoryPostProcessor

* 扩展原理:* BeanPostProcessor:bean后置处理器,bean创建对象初始化前后进行拦截工作的** 1、BeanFactoryPostProcessor:beanFactory的后置处理器;* 在BeanFactory标准初始化之后调用,来定制和修改BeanFactory的内容;* 所有的bean定义已经保存加载到beanFactory,但是bean的实例还未创建*** BeanFactoryPostProcessor原理:* 1)、ioc容器创建对象* 2)、invokeBeanFactoryPostProcessors(beanFactory);* 如何找到所有的BeanFactoryPostProcessor并执行他们的方法;* 1)、直接在BeanFactory中找到所有类型是BeanFactoryPostProcessor的组件,并执行他们的方法* 2)、在初始化创建其他组件前面执行

代码实现 

@Componentpublic 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('当前BeanFactory中有'+count+' 个Bean'); System.out.println(Arrays.asList(names)); } }

2、扩展原理-BeanDefinitionRegistryPostProcessor

BeanDefinitionRegistryPostProcessor

* 2、BeanDefinitionRegistryPostProcessor extends BeanFactoryPostProcessor* postProcessBeanDefinitionRegistry();* 在所有bean定义信息将要被加载,bean实例还未创建的;** 优先于BeanFactoryPostProcessor执行;* 利用BeanDefinitionRegistryPostProcessor给容器中再额外添加一些组件;** 原理:* 1)、ioc创建对象* 2)、refresh()-》invokeBeanFactoryPostProcessors(beanFactory);* 3)、从容器中获取到所有的BeanDefinitionRegistryPostProcessor组件。* 1、依次触发所有的postProcessBeanDefinitionRegistry()方法* 2、再来触发postProcessBeanFactory()方法BeanFactoryPostProcessor;** 4)、再来从容器中找到BeanFactoryPostProcessor组件;然后依次触发postProcessBeanFactory()方法

代码实现

@Componentpublic class MyBeanDefinitionRegistryPostProcessor implements BeanDefinitionRegistryPostProcessor{ public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException { // TODO Auto-generated method stub System.out.println('MyBeanDefinitionRegistryPostProcessor...bean的数量:'+beanFactory.getBeanDefinitionCount()); } //BeanDefinitionRegistry Bean定义信息的保存中心,以后BeanFactory就是按照BeanDefinitionRegistry里面保存的每一个bean定义信息创建bean实例; public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException { // TODO Auto-generated method stub System.out.println('postProcessBeanDefinitionRegistry...bean的数量:'+registry.getBeanDefinitionCount()); //RootBeanDefinition beanDefinition = new RootBeanDefinition(Blue.class); AbstractBeanDefinition beanDefinition = BeanDefinitionBuilder.rootBeanDefinition(Blue.class).getBeanDefinition(); registry.registerBeanDefinition('hello', beanDefinition); } }

3、扩展原理-ApplicationListener用法

监听器ApplicationListener

* 3、ApplicationListener:监听容器中发布的事件。事件驱动模型开发;* public interface ApplicationListener<E extends ApplicationEvent>* 监听 ApplicationEvent 及其下面的子事件;** 步骤:* 1)、写一个监听器(ApplicationListener实现类)来监听某个事件(ApplicationEvent及其子类)* @EventListener;* 原理:使用EventListenerMethodProcessor处理器来解析方法上的@EventListener;** 2)、把监听器加入到容器;* 3)、只要容器中有相关事件的发布,我们就能监听到这个事件;* ContextRefreshedEvent:容器刷新完成(所有bean都完全创建)会发布这个事件;* ContextClosedEvent:关闭容器会发布这个事件;* 4)、发布一个事件:* applicationContext.publishEvent();

代码实现:

方式一:实现ApplicationListener<E extends ApplicationEvent>接口

@Componentpublic class MyApplicationListener implements ApplicationListener<ApplicationEvent> { //当容器中发布此事件以后,方法触发 public void onApplicationEvent(ApplicationEvent event) { // TODO Auto-generated method stub System.out.println('收到事件:'+event); }}

方式二:使用@EventListener注解标识事件监听方法

@Servicepublic class UserService { @EventListener(classes={ApplicationEvent.class}) public void listen(ApplicationEvent event){ System.out.println('UserService。。监听到的事件:'+event); }}

4、扩展原理-ApplicationListener原理

* 原理:* ContextRefreshedEvent、IOCTest_Ext$1[source=我发布的时间]、ContextClosedEvent;* 1)、ContextRefreshedEvent事件:* 1)、容器创建对象:refresh();* 2)、finishRefresh();容器刷新完成会发布ContextRefreshedEvent事件* 2)、自己发布事件;* 3)、容器关闭会发布ContextClosedEvent;* * 【事件发布流程】:* 3)、publishEvent(new ContextRefreshedEvent(this));* 1)、获取事件的多播器(派发器):getApplicationEventMulticaster()* 2)、multicastEvent派发事件:* 3)、获取到所有的ApplicationListener;* for (final ApplicationListener<?> listener : getApplicationListeners(event, type)) {* 1)、如果有Executor,可以支持使用Executor进行异步派发;* Executor executor = getTaskExecutor();* 2)、否则,同步的方式直接执行listener方法;invokeListener(listener, event);* 拿到listener回调onApplicationEvent方法;* * 【事件多播器(派发器)】* 1)、容器创建对象:refresh();* 2)、initApplicationEventMulticaster();初始化ApplicationEventMulticaster;* 1)、先去容器中找有没有id=“applicationEventMulticaster”的组件;* 2)、如果没有this.applicationEventMulticaster = new SimpleApplicationEventMulticaster(beanFactory);* 并且加入到容器中,我们就可以在其他组件要派发事件,自动注入这个applicationEventMulticaster;* * 【容器中有哪些监听器】* 1)、容器创建对象:refresh();* 2)、注册监听器:registerListeners();* 从容器中拿到所有的监听器,把他们注册到applicationEventMulticaster中;* String[] listenerBeanNames = getBeanNamesForType(ApplicationListener.class, true, false);* //将listener注册到ApplicationEventMulticaster中* getApplicationEventMulticaster().addApplicationListenerBean(listenerBeanName);

5、扩展原理-@EventListener与SmartInitializingSingleton

* SmartInitializingSingleton 原理:->afterSingletonsInstantiated();* 1)、ioc容器创建对象并refresh();* 2)、finishBeanFactoryInitialization(beanFactory);初始化剩下的单实例bean;* 1)、先创建所有的单实例bean;getBean();* 2)、获取所有创建好的单实例bean,判断是否是SmartInitializingSingleton类型的;* 如果是就调用afterSingletonsInstantiated();

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持好吧啦网。

标签: Spring
相关文章: