zoukankan      html  css  js  c++  java
  • Spring注解驱动开发(五)-----扩展原理

    扩展原理

    1、BeanPostProcessor-----bean后置处理器,bean创建对象初始化前后进行拦截工作的

    2、BeanFactoryPostProcessor-----beanFactory的后置处理器
    在BeanFactory标准初始化之后调用,来定制和修改BeanFactory的内容;
    所有的bean定义已经保存加载到beanFactory,但是bean的实例还未创建。

    :首先spring容器会创建beanDefinition,此时bean并没有初始化。

    示例:

    ExtConfig:

    @ComponentScan("com.atguigu.ext")
    @Configuration
    public class ExtConfig {
        
        @Bean
        public Blue blue(){
            return new Blue();
        }
    }

    Blue:

    package com.atguigu.bean;
    
    public class Blue {
        
        public Blue(){
            System.out.println("blue...constructor");
        }
        
        public void init(){
            System.out.println("blue...init...");
        }
        
        public void detory(){
            System.out.println("blue...detory...");
        }
    }

    MyBeanFactoryPostProcessor:

    package com.atguigu.ext;
    
    import java.util.Arrays;
    
    import org.springframework.beans.BeansException;
    import org.springframework.beans.factory.config.BeanFactoryPostProcessor;
    import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;
    import org.springframework.stereotype.Component;
    
    @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));
        }
    }

    IOCTest_Ext:

    package com.atguigu.test;
    
    import org.junit.Test;
    import org.springframework.context.ApplicationEvent;
    import org.springframework.context.annotation.AnnotationConfigApplicationContext;
    
    import com.atguigu.aop.MathCalculator;
    import com.atguigu.bean.Boss;
    import com.atguigu.bean.Car;
    import com.atguigu.bean.Color;
    import com.atguigu.bean.Red;
    import com.atguigu.config.MainConfigOfAOP;
    import com.atguigu.config.MainConifgOfAutowired;
    import com.atguigu.dao.BookDao;
    import com.atguigu.ext.ExtConfig;
    import com.atguigu.service.BookService;
    
    public class IOCTest_Ext {
        
        @Test
        public void test01(){
            AnnotationConfigApplicationContext applicationContext  = new AnnotationConfigApplicationContext(ExtConfig.class);              
            applicationContext.close();
        }
    }

    运行结果:

    在blue对象初始化之前,做点工作!~~~

    BeanFactoryPostProcessor原理

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

    BeanDefinitionRegistryPostProcessor 

    BeanDefinitionRegistryPostProcessor extends BeanFactoryPostProcessor 

      postProcessBeanDefinitionRegistry();
      在所有bean定义信息将要被加载,bean实例还未创建的;
      优先于BeanFactoryPostProcessor执行;
      利用BeanDefinitionRegistryPostProcessor给容器中再额外添加一些组件;

    MyBeanDefinitionRegistryPostProcessor:

    package com.atguigu.ext;
    
    import org.springframework.beans.BeansException;
    import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;
    import org.springframework.beans.factory.support.AbstractBeanDefinition;
    import org.springframework.beans.factory.support.BeanDefinitionBuilder;
    import org.springframework.beans.factory.support.BeanDefinitionRegistry;
    import org.springframework.beans.factory.support.BeanDefinitionRegistryPostProcessor;
    import org.springframework.beans.factory.support.RootBeanDefinition;
    import org.springframework.stereotype.Component;
    
    import com.atguigu.bean.Blue;
    
    @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 

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

    public interface ApplicationListener<E extends ApplicationEvent>

    监听 ApplicationEvent 及其下面的子事件;

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

    MyApplicationListener:

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

    IOCTest_Ext:

    package com.atguigu.test;
    
    import org.junit.Test;
    import org.springframework.context.ApplicationEvent;
    import org.springframework.context.annotation.AnnotationConfigApplicationContext;
    
    import com.atguigu.aop.MathCalculator;
    import com.atguigu.bean.Boss;
    import com.atguigu.bean.Car;
    import com.atguigu.bean.Color;
    import com.atguigu.bean.Red;
    import com.atguigu.config.MainConfigOfAOP;
    import com.atguigu.config.MainConifgOfAutowired;
    import com.atguigu.dao.BookDao;
    import com.atguigu.ext.ExtConfig;
    import com.atguigu.service.BookService;
    
    public class IOCTest_Ext {
        
        @Test
        public void test01(){
            AnnotationConfigApplicationContext applicationContext  = new AnnotationConfigApplicationContext(ExtConfig.class);
            
            
            //发布事件;
            applicationContext.publishEvent(new ApplicationEvent(new String("我发布的事件")) {
            });
            
            applicationContext.close();
        }
    
    }

    也可以在方法上使用注解的方式来定义监听器-----@EventListener后定义需要监听的事件

    package com.atguigu.ext;
    
    import org.springframework.context.ApplicationEvent;
    import org.springframework.context.event.EventListener;
    import org.springframework.stereotype.Service;
    
    @Service
    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;

    【事件多播器(派发器)】
    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);

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

  • 相关阅读:
    CF251D
    P6914
    CF1100F
    双连通 / 圆方树 胡扯笔记
    P4082
    SparkSql使用Hive中注册的UDF函数报类找不到问题解决
    Oracle 查询时使用时间作为where报错hour must be between 1 and 12
    【面试-python】
    Linux和Git
    AMBA初探
  • 原文地址:https://www.cnblogs.com/alimayun/p/11123716.html
Copyright © 2011-2022 走看看