zoukankan      html  css  js  c++  java
  • 20、spring注解学习(扩展原理)——扩展原理

    1、扩展原理-BeanFactoryPostProcessor

    BeanFactoryPostProcessor

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

    代码实现

    @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("当前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()方法

    代码实现

    @Component
    public 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>接口

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

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

    @Service
    public 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);

    扩展原理-@EventListener与SmartInitializingSingleton

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

    摘自:https://www.cnblogs.com/wjqhuaxia/p/12261814.html

  • 相关阅读:
    JS中使用正则表达式封装的一些常用的格式验证的方法-是否外部url、是否小写、邮箱格式、是否字符、是否数组
    Java中操作字符串的工具类-判空、截取、格式化、转换驼峰、转集合和list、是否包含
    Cocos2d-x 2.0 自适应多种分辨率
    应用自定义移动设备外观
    为移动设备应用程序创建外观
    【2020-11-28】人生十三信条
    【2020-11-27】事实证明,逃避是下等策略
    Python 之web动态服务器
    Python 之pygame飞机游戏
    PHP 之转换excel表格中的经纬度
  • 原文地址:https://www.cnblogs.com/lyh233/p/12469057.html
Copyright © 2011-2022 走看看