Spring注解开发之扩展原理

Spring注解开发之扩展原理

1、BeanFactoryPostProcessor

beanFactory的后置处理器

  • 在BeanFactory标准初始化之后调用,来定制和修改BeanFactory的内容;

  • 所有的bean定义已经保存加载到beanFactory,但是bean的实例还未创建

  • 注意:和BeanPostProcessor的区别:

    • BeanPostProcessor是bean后置处理器,bean创建对象初始化前后进行拦截工作的
@Component
public class MyBeanFactoryPostProcessor implements BeanFactoryPostProcessor {

	@Override
	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));
	}

}

原理:

  • 1)、ioc容器创建对象
  • 2)、invokeBeanFactoryPostProcessors(beanFactory);
    • 如何找到所有的BeanFactoryPostProcessor并执行他们的方法;
      • 1)、直接在BeanFactory中找到所有类型是BeanFactoryPostProcessor的组件,并执行他们的方法
      • 2)、在初始化创建其他组件前面执行

2、BeanDefinitionRegistryPostProcessor

  • 它是BeanFactoryPostProcessor的子接口,增加了方法postProcessBeanDefinitionRegistry();
  • 在所有bean定义信息将要被加载,bean实例还未创建的;
  • 优先于BeanFactoryPostProcessor执行;
  • 可以利用BeanDefinitionRegistryPostProcessor给容器中再额外添加一些组件;
@Component
public class MyBeanDefinitionRegistryPostProcessor implements BeanDefinitionRegistryPostProcessor{

   @Override
   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实例;
   @Override
   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);
   }

原理:

  • 1)、ioc创建对象
  • 2)、refresh()-》invokeBeanFactoryPostProcessors(beanFactory);
  • 3)、从容器中获取到所有的BeanDefinitionRegistryPostProcessor组件。
    • 1、依次触发所有的postProcessBeanDefinitionRegistry()方法
    • 2、再来触发postProcessBeanFactory()方法(BeanFactoryPostProcessor的方法)
  • 4)、再来从容器中找到BeanFactoryPostProcessor组件;然后依次触发postProcessBeanFactory()方法

3、ApplicationListener

  • 监听容器中发布的事件----事件驱动模型开发

    public interface ApplicationListener

  • 负责监听 ApplicationEvent 及其下面的子事件

  • 步骤:

    • 1)、写一个监听器(ApplicationListener实现类)来监听某个事件(ApplicationEvent及其子类)

      或者在方法上标注@EventListener;

    • 2)、把监听器加入到容器;

    • 3)、只要容器中有相关事件的发布,我们就能监听到这个事件;

    • ContextRefreshedEvent:容器刷新完成(所有bean都完全创建)会发布这个事件;

    • ContextClosedEvent:关闭容器会发布这个事件;

  • 4)、发布一个事件:

    • applicationContext.publishEvent();
@Component
public class MyApplicationListener implements ApplicationListener<ApplicationEvent> {

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

}
public class IOCTest_Ext {
   
   @Test
   public void test01(){
      AnnotationConfigApplicationContext applicationContext  = new AnnotationConfigApplicationContext(ExtConfig.class);
      
      
      //发布事件;
      applicationContext.publishEvent(new ApplicationEvent(new String("我发布的事件")) {
      });
      
      applicationContext.close();
   }

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

}

原理:

  • 三个事件:ContextRefreshedEvent、IOCTest_Ext$1[source=我发布的事件]、ContextClosedEvent;

    • 1)、ContextRefreshedEvent事件:
      • 1)、容器创建对象:refresh();
      • 2)、finishRefresh();容器刷新完成会发布ContextRefreshedEvent事件
    • 2)、自己发布事件;
    • 3)、容器关闭会发布ContextClosedEvent;
    • 注意:不管哪个事件,都会调用到publishEvent方法
      • 【事件发布流程】:
        • 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);
  • @Listener原理

    • 使用EventListenerMethodProcessor处理器来解析方法上的@EventListener;
    • SmartInitializingSingleton 原理:->afterSingletonsInstantiated();
      • 1)、ioc容器创建对象并refresh();
      • 2)、finishBeanFactoryInitialization(beanFactory);初始化剩下的单实例bean;
        • 1)、先创建所有的单实例bean;getBean();
        • 2)、获取所有创建好的单实例bean,判断是否是SmartInitializingSingleton类型的;
          •  			如果是就调用afterSingletonsInstantiated();
            
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值