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

  1、BeanFactoryPostProcessor:BeanFactory的后置处理器;

  在BeanFactory标准初始化之后调用;所有的bean定义已经保存加载到BeanFactory,但是bean的实例还没创建;

  BeanFactoryPostProcessor原理:

  1.1 ioc容器创建对象

  1.2 invokeBeanFactoryPostProcessors(BeanFactory); 执行BeanFactoryPostProcessor;

  如何找到所有的BeanFactoryPostProcessor并执行他们的方法;

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

  1.2.2 在初始化创建其他组件前面执行

  2、BeanDefinitionRegistryPostProcessor extends BeanFactoryPostProcessor

  postProcessBeanDefinitionRegistry();

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

  优先于BeanFactoryPostProcessor执行;

  利用BeanDefinitionRegistryPostProcessor给容器中在额外添加一些组件;

  原理:

  2.1、ioc创建对象

  2.2、refresh()->invokeBeanFactoryPostProcessors(beanFactory);

  2.3、从容器中获取到的BeanDefinitionRegistryPostProcessor组件。

  2.3.1 依次触发所有的postProcessBeanDefinitionRegistry()方法

  2.3.2 再来触发postProcessBeanFactory()方法BeanFactoryPostProcessor;

  2.4、再来从容器中找到BeanFactoryPostProcessor组件;然后依次触发postProcessBeanFactory()方法

  3、ApplicationListener:监听容器中发布的事件。事件驱动模型开发;

  public interface ApplicationListener

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

  步骤:

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

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

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

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

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

  4、发布一个事件;

  applicationContext.publishEvent();

  原理:

  ContextRefreshedEvent、IOCTest_Ext$1[source=自己的事件!!]、ContextClosedEvent;

  1、ContextRefreshedEvent事件:

  1.1 容器创建对象:refresh();

  1.2 finishRefresh();容器刷新完成会发布ContextRefreshedEvent

  1.3 publishEvent(new ContextRefreshedEvent(this));【事件发布流程】

  1.3.1 获取事件的多波器(派发器):getApplicationEventMulticaster()

  1.3.2 MulticasterEvent派发事件:

  1.3.3 获取到所有的ApplicationListener

  for(final ApplicationListener listener : getApplicationListeners(event, type))

  1.3.3.1 如果有Executor,可以支持使用Executor进行异步派发;

  Executor executor = getTaskExecutor();

  1.3.3.2 否则,同步的方式直接执行listener方法;invokeListener(listener, event);

  拿到listener回调onApplicationEvent方法;

  2、发布自己的事件;

  3、容器关闭会发布ContextClosedEvent

  【事件多波器(派发器)】

  1、容器创建对象:refresh();

  2、initApplicationEventMulticaster();初始化ApplicationEventMulticaster;

  2.1 先去容器中找到有没有id="applicationEventMultcaster"的组件;

  2.2 如果没有this.applicationEventMulticaster = new SimpleApplicationEventMulticaster(beanFactory);

  并且加入到容器中,我们就可以在其他组件要派发事件,自动注入这个applicationEventMulticaster;

  【容器中有哪些监听器】

  1、容器创建对象:refresh();

  2、registerListeners();

  String[] listenerBeanNames = getBeanNamesForType(ApplicationListener.class, true, false);

  //将listener注册到ApplicationEventMulticaster中

  getApplicationEventMulticaster().addApplicationListenerBean(listenerBeanName);

  package com.spring.ext;

  import org.springframework.context.annotation.Bean;

  import org.springframework.context.annotation.ComponentScan;

  import org.springframework.context.annotation.Configuration;

  import com.spring.bean.Red;

  @ComponentScan("com.spring.ext")

  @Configuration

  public class ExtConfig {

  @Bean

  public Red blue() {

  return new Red();

  }

  }

  package com.spring.ext;

  import org.springframework.context.ApplicationEvent;

  import org.springframework.context.ApplicationListener;

  import org.springframework.stereotype.Component;

  @Component

  public class MyApplicationListener implements ApplicationListener{

  public void onApplicationEvent(ApplicationEvent event) {

  System.out.println("event>>>>>>>>"+event);

  }

  }

  package com.spring.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 {

  public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {

  System.out.println("MyBeanFactoryPostProcessor ... postProcessBeanFactory beansize="+beanFactory.getBeanDefinitionCount());

  int count = beanFactory.getBeanDefinitionCount();

  String[] names = beanFactory.getBeanDefinitionNames();

  System.out.println("当前BeanFactory中有"+count+"个Bean");

  System.out.println(Arrays.toString(names));

  }无锡妇科医院哪家好 http://www.ytsgfk120.com/

  }

  package com.spring.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.stereotype.Component;

  import com.spring.bean.Red;

  @Component

  public class MyBenaDefinitionRegistryPostProcessor implements BeanDefinitionRegistryPostProcessor {

  public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {

  System.out.println("MyBenaDefinitionRegistryPostProcessor ..beansize ="+beanFactory.getBeanDefinitionCount());

  }

  public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException {

  System.out.println(">>>>>postProcessBeanDefinitionRegistry>>>beansize="+registry.getBeanDefinitionCount());

  AbstractBeanDefinition beanDefinition = BeanDefinitionBuilder.rootBeanDefinition(Red.class).getBeanDefinition();

  registry.registerBeanDefinition("hello", beanDefinition);

  }

  }

  package com.spring.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 listener(ApplicationEvent event) {

  System.out.println("自己监听><>>>>>>"+event);

  }

  }

  package com.spring.test;

  import org.junit.Test;

  import org.springframework.context.ApplicationEvent;

  import org.springframework.context.annotation.AnnotationConfigApplicationContext;

  import com.spring.ext.ExtConfig;

  public class IOCTestOfExt {

  @Test

  public void test01() {

  AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(ExtConfig.class);

  applicationContext.publishEvent(new ApplicationEvent("自己的事件!!") {

  });

  applicationContext.close();

  }

  }

BeanPostProcessor后置处理器原理以及ApplicationListener原理的更多相关文章

  1. 【Spring注解驱动开发】关于BeanPostProcessor后置处理器,你了解多少?

    写在前面 有些小伙伴问我,学习Spring是不是不用学习到这么细节的程度啊?感觉这些细节的部分在实际工作中使用不到啊,我到底需不需要学习到这么细节的程度呢?我的答案是:有必要学习到这么细节的程度,而且 ...

  2. Spring Ioc源码分析系列--Ioc容器注册BeanPostProcessor后置处理器以及事件消息处理

    Spring Ioc源码分析系列--Ioc容器注册BeanPostProcessor后置处理器以及事件消息处理 前言 上一篇分析了BeanFactoryPostProcessor的作用,那么这一篇继续 ...

  3. Spring之BeanPostProcessor(后置处理器)介绍

      为了弄清楚Spring框架,我们需要分别弄清楚相关核心接口的作用,本文来介绍下BeanPostProcessor接口 BeanPostProcessor   该接口我们也叫后置处理器,作用是在Be ...

  4. Spring的BeanPostProcessor后置处理器与bean的生命周期

    前言 本文将把Spring在Bean的生命周期中涉及到的后置处理器一一梳理出来,并简要说一下功能,至于每个后置处理器在实际扩展中的用处,还要后续慢慢探索总结. 正文 下面一步步跟进探寻那些后置处理器们 ...

  5. 1.spring源码-BeanPostProcessor后置处理器

    1.BeanPostProcessor接口的介绍: BeanPostProcessor是一个接口,其中有两个方法,postProcessBeforeInitialization和postProcess ...

  6. Spring点滴五:Spring中的后置处理器BeanPostProcessor讲解

    BeanPostProcessor接口作用: 如果我们想在Spring容器中完成bean实例化.配置以及其他初始化方法前后要添加一些自己逻辑处理.我们需要定义一个或多个BeanPostProcesso ...

  7. spring学习四:Spring中的后置处理器BeanPostProcessor

    BeanPostProcessor接口作用: 如果我们想在Spring容器中完成bean实例化.配置以及其他初始化方法前后要添加一些自己逻辑处理.我们需要定义一个或多个BeanPostProcesso ...

  8. 2.spring源码-BeanPostProcessor后置处理之ApplicationContextAwareProcessor,实现spring容器中某一个类的bean对象在初始化时需要得到Spring容器内容。

    需求:我们的需求是,在spring初始化完毕时,使我们自定义一个类Bird类可以得到spring容器内容. 实现步骤: 1.首先我们来看一下ApplicationContextAwareProcess ...

  9. spring的后置处理器——BeanPostProcessor以及spring的生命周期

    后置处理器的调用时机 BeanPostProcessor是spring提供的接口,它有两个方法——postProcessBeforeInitialization.postProcessAfterIni ...

随机推荐

  1. mybatis框架的分页功能

    需求说明:为用户管理之查询用户列表功能增加分页实现      列表结果按照创建时间降序排列 /** * 需求说明:为用户管理之查询用户列表功能增加分页实现 列表结果按照创建时间降序排列 * @para ...

  2. ZJOI2019赛季回顾

    退役了. NOIP2018 day1没什么好说的. day2开考后看完题:这个T3 TM不是DDP吗? 考前刚学过这东西,还没去写过 当时不知道在想什么,胡了T1 60和T2 50分保底之后就去刚T3 ...

  3. mysql数据库锁的机制-及事务事件

    事务隔离级别,脏读.不可重复读.幻读,乐观锁.悲观锁(共享锁.排它锁) 数据库事务具有四个特征,分别是原子性(Atomicity).一致性(Consistency).隔离性(Isoation).持久性 ...

  4. [考试]NOIP2015模拟题2

    // 此博文为迁移而来,写于2015年7月22日,不代表本人现在的观点与看法.原始地址:http://blog.sina.com.cn/s/blog_6022c4720102w72i.html 1.总 ...

  5. UID、PID、PPID是什么?

    UID是用户ID,PID是进程ID,PPID是父进程ID. UID UID 用户身份证明(User Identification)的缩写.UID用户在注册后,系统会自动的给你一个UID的数值.意思就是 ...

  6. rpm安装和卸载

    以jdk的卸载和安装为例. 1. 查找已安装的包名 $ rpm -q jdk jdk-1.7.0_79-fcs.x86_64 如果系统中有已经安装的jdk包,则会输出包名. 2. 卸载已安装的包 $ ...

  7. 日常运维--rsync同步工具

    rsync命令是一个远程数据同步工具,可通过LAN/WAN快速同步多台主机间的文件.rsync使用所谓的“rsync算法”来使本地和远程两个主机之间的文件达到同步,这个算法只传送两个文件的不同部分,而 ...

  8. 一文带你了解elasticsearch

    一文带你了解elasticsearch cxf2102100人评论160人阅读2019-07-02 21:31:36   elasticsearch es基本概念 es术语介绍 文档Document ...

  9. Elasticsearch详解-续

    Elasticsearch详解-续 Chandler_珏瑜  关注  7.6 2019.05.22 10:46* 字数 8366 阅读 675评论 4喜欢 25 5.3 性能调优  Elasticse ...

  10. 【RabbitMQ学习之一】RabbitMQ入门

    环境 win7 rabbitmq-server-3.7.17 Erlang 22.1 RabbitMQ使用Erlang语言开发消息中间件.RabbitMQ基于AMQP(高级消息队列协议)协议,更适合业 ...