zoukankan      html  css  js  c++  java
  • Spring注解驱动开发之扩展原理

    前言:现今SpringBoot、SpringCloud技术非常火热,作为Spring之上的框架,他们大量使用到了Spring的一些底层注解、原理,比如@Conditional、@Import、@EnableXXX等。如果掌握这些底层原理、注解,那么我们对这些高层框架就能做到高度定制,使用的游刃有余

    一、BeanFactoryPostProcessor

      BeanPostProcessor是bean后置处理器,bean创建对象初始化前后进行拦截工作的,而BeanFactoryPostProcessor是beanFactory的后置处理器在BeanFactory标准初始化之后调用,来定制和修改BeanFactory的内容所有的bean定义已经保存加载到beanFactory,但是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));
        }
    }

    BeanFactoryPostProcessor原理:  

      1)、ioc容器创建对象

      2)、invokeBeanFactoryPostProcessors(beanFactory)

        1)、直接在BeanFactory中找到所有类型是BeanFactoryPostProcessor的组件,并执行他们的方法

        2)、在初始化创建其他组件前面执行

    二、BeanDefinitionRegistryPostProcessor

      BeanDefinitionRegistryPostProcessor 继承于 BeanFactoryPostProcessor,postProcessBeanDefinitionRegistry()方法:

        1)、在所有bean定义信息将要被加载,bean实例还未创建的;

        2)、优先于BeanFactoryPostProcessor执行;

        3)、利用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()方法
      
      

    三、ApplicationListener 

    1.使用:

    @Component
    public class MyApplicationListener implements ApplicationListener<ApplicationEvent> {
    
        //当容器中发布此事件以后,方法触发
        @Override
        public void onApplicationEvent(ApplicationEvent event) {
            // TODO Auto-generated method stub
            System.out.println("收到事件:"+event);
        }
    }
    @Service
    public class UserService {
        
        @EventListener(classes={ApplicationEvent.class})
        public void listen(ApplicationEvent event){
            System.out.println("UserService。。监听到的事件:"+event);
        }
    
    }

    通过实现ApplicationListener(在方法上标注@EventListener)来监听容器中发布的事件(ApplicationEvent 及其下面的子事件):

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

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

      发布一个事件:applicationContext.publishEvent()

    2、原理:

      1)、ContextRefreshedEvent事件:
        1)、容器创建对象:refresh();
        2)、finishRefresh();容器刷新完成会发布ContextRefreshedEvent事件
      2)、自己发布事件;
      3)、容器关闭会发布ContextClosedEvent;

    【事件发布流程】:
      4)、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);

    @EventListener原理:使用EventListenerMethodProcessor处理器来解析方法上的@EventListener:

      EventListenerMethodProcessor实现了SmartInitializingSingleton,afterSingletonsInstantiated()执行原理:

        1)、ioc容器创建对象并refresh()

        2)、finishBeanFactoryInitialization(beanFactory);初始化剩下的单实例bean

          1)、先创建所有的单实例bean;getBean()

          2)、获取所有创建好的单实例bean,判断是否是SmartInitializingSingleton类型的,如果是就调用afterSingletonsInstantiated()

      

  • 相关阅读:
    Python3 collections模块的使用
    基于python的分治法和例题
    docker容器间通信 (共用宿主机)
    HTML之form表单ENCTYPE属性解析
    搭建基于码云gitee平台代码自动部署
    centos7下docker搭建nginx+phpfpm环境
    mysql主从配置
    centos7升级自带mariadb
    linux下安装docker
    centos7安装postgreSql11
  • 原文地址:https://www.cnblogs.com/qzlcl/p/11006992.html
Copyright © 2011-2022 走看看