/**
* @Author quan
* @Date 2020/11/13
* 扩展原理
* BeanPostProcessor bean后置处理器,bean创建对象初始化前后进行拦截工作
*
*
* BeanFactoryPostProcessor BeanFactory后置处理器
* 在BeanFactory标准初始化之后调用,所有的Bean定义已经保存加载到BeanFactory,可以修改订制BeanFactory的内容
* 但是BeanFactory还没帮我们实例化
*
* 1ioc容器创建对象
* 2invokeBeanFactoryPostProcessors(beanFactory)执行BeanFactoryPostProcessor
* 如何拿到所有BeanFactoryPostProcessors 在这个类PostProcessorRegistrationDelegate(invoker入口AbstractApplicationContext)
* String[] postProcessorNames =
* beanFactory.getBeanNamesForType(BeanFactoryPostProcessor.class, true, false);
* 这里取获取所有后置bean的name,然后进行排序
* 1.直接在BeanFactory中找到所有类型是BeanFactoryPostProcessor的组件,并执行他们的方法
* 2.在初始化创建其他组件前执行
*
*
* BeanDefinitionRegistryPostProcessor extends BeanFactoryPostProcessor
* void postProcessBeanDefinitionRegistry
* 指定世纪:看源码注解iiu知道了。在所有Bean定义信息将要被加载的时候,bean实例还未创建的时候
* 比BeanFactoryPostProcessor还要前,
* 利用BeanDefinitionRegistryPostProcessor给容器中添加额外添加一些组件
*
* 原理:
* 1ioc容器创建
* 2refresh()->invokeBeanFactoryPostProcessors(beanFactory);
* 3从容器中获取到所有的BeanDefinitionRegistryPostProcessor AbstractApplicationContext#invokeBeanFactoryPostProcessors实际上执行:PostProcessorRegistrationDelegate.invokeBeanFactoryPostProcessors
* 4依次触发所有的postProcessBeanDefinitionRegistry invokeBeanDefinitionRegistryPostProcessors(currentRegistryProcessors, registry, beanFactory.getApplicationStartup());
* 5再触发:postProcessBeanFactory实际上是再BeanFactoryPostProcessor
* 6再从容器中找到BeanFactoryPostProcessor组件,返回一次触发postProcessBeanFactory。
*
*
* ApplicationListener:监听容器中发布的时间,事件驱动模型开发
* public interface ApplicationListener<E extends ApplicationEvent> extends EventListener
* 都要监听ApplicationEvent及其下面的子事件:
* 步骤:
* 1下一个监听器来监听某个事件(ApplicationEvent及其子类)
* 2监听器加入到容器
* 3只要容器中有相关事件的发布,我们九年监听到这个事件。
* Spring发布的事件 ContextRefreshedEvent 容器刷新完成(所有Bean都被完全创建),会发布这个事件
* ContextClosedEvent 关闭容器会发布这个事件
* 4发布一个事件:
* applicationContext.publishEvent();
*
* 原理:
* 第一个事件:ContextRefreshedEvent
* 1容器创建 refresh()
* 2finishRefresh容器刷新完成
* 这个方法里面调用了 publishEvent(new ContextRefreshedEvent(this))
* 里面发布流程:
* 1-获取事件的多播器()getApplicationEventMulticaster()
* 2-multicastEvent派发事件
* 3-获取所有的ApplicationListener
* for (ApplicationListener<?> listener : getApplicationListeners(event, type)) {
* 1-如果有Executor 可以支持使用Executor进行异步派发
* Executor executor = getTaskExecutor();
* 2-否则同步执行invokeListener(listener, event);
* 3-拿到Listener回调onApplicationEvent方法listener.onApplicationEvent(event)
* 最后一个事件:ContextCloseEvent
* AbstractApplicationContext#doClose();
*
*
* {事件多播器怎么获取的呢}:
* 1.容器创建对象Refresh:
* 2.AbstractApplicationContext#initApplicationEventMulticaster();初始化事件多播器
* 1-先去BeanFactory里面有没有id=applicationEventMulticaster的Bean
* 2-如果没有就新建一个new SimpleApplicationEventMulticaster(beanFactory),并注册进去:
* beanFactory.registerSingleton(APPLICATION_EVENT_MULTICASTER_BEAN_NAME, this.applicationEventMulticaster);
* 那就可以再其他组件要派发事件,则自动注入即可
*
* {容器有那些监听器}
* 1.容器创建对象Refresh:
* 2.registerListeners();
* 从容器中拿到所有的监听器,
* String[] listenerBeanNames = getBeanNamesForType(ApplicationListener.class, true, false);
* 把他们注册到ApplicationEventMulticaster中
* getApplicationEventMulticaster().addApplicationListenerBean(listenerBeanName);
*
* #另一种是心啊监听器的方法@EventListener
* 原理:
* 1-点进注解里面,看注释可以直到主要工作的市EventListenerMethodProcessor来解析注解
*
* SmartInitializingSingleton原理:---》执行时机afterSingletonsInstantiated
* 1-ioc容器创建对象并refresh
* 2-finishBeanFactoryInitialization 初始化剩下的单实例bean
* 1-其实执行的市DefaultListableBeanFactory#preInstantiateSingletons
* 2-先创建所有的单实例bean,getBean
* 3-获取所有创建好的单实例bean,判断是否是SmartInitializingSingleton类型
* 并执行smartSingleton.afterSingletonsInstantiated();
*/
/**
* @Author quan
* @Date 2020/11/13
* 扩展点=-BeanFactoryPostProcessor
* <p>
* <p>
* * Modify the application context's internal bean factory after its standard
* * initialization. All bean definitions will have been loaded, but no beans
* * will have been instantiated yet. This allows for overriding or adding
* * properties even to eager-initializing beans.
* * @param beanFactory the bean factory used by the application context
* * @throws org.springframework.beans.BeansException in case of errors
*/ @Component
public class MyBeanFactoryPostProcessor implements BeanFactoryPostProcessor {
@Override
public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
// System.out.println(" MyBeanFactoryPostProcessor");
int count = beanFactory.getBeanDefinitionCount();
System.out.println("MyBeanFactoryPostProcessor"+count);
String[] names = beanFactory.getBeanDefinitionNames();
System.out.println(Arrays.asList(names));
}
}
/**
* @Author quan
* @Date 2020/11/13
*/
@Component
public class MyBeanDefinitionRegistryPostProcessor implements BeanDefinitionRegistryPostProcessor { //BeanDefinitionRegistry Bean定义信息的保存中心
//BeanFactory都是按照BeanDefinitionRegistry里面保存的每一个Bean定义信息创建Bean的实例
@Override
public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException {
System.out.println("MyBeanDefinitionRegistryPostProcessor-postProcessBeanDefinitionRegistry数量" +registry.getBeanDefinitionCount());
// RootBeanDefinition beanDefinition = new RootBeanDefinition(Quan.class);
AbstractBeanDefinition beanDefinition = BeanDefinitionBuilder.rootBeanDefinition(Quan.class).getBeanDefinition();
registry.registerBeanDefinition("quan",beanDefinition);
} @Override
public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
System.out.println("MyBeanDefinitionRegistryPostProcessor-postProcessBeanFactory数量" +beanFactory.getBeanDefinitionCount());
}
}
/**
* @Author quan
* @Date 2020/11/20
*/
@Service
public class MyServiceEventListener { @EventListener(classes = {ApplicationEvent.class})
public void listen(ApplicationEvent event){
System.out.println("EventListener"+event);
}
}
/**
* @Author quan
* @Date 2020/11/18
*/
@Component//需要加载到容器中去
public class MyApplicationListener implements ApplicationListener<ApplicationEvent> {
//当容器发布此事件的时候,方法会得到促发
@Override
public void onApplicationEvent(ApplicationEvent event) {
System.out.println("事件-----"+event);
}
}

spring源码-扩展点的更多相关文章

  1. Spring源码系列 — BeanDefinition扩展点

    前言 前文介绍了Spring Bean的生命周期,也算是XML IOC系列的完结.但是Spring的博大精深,还有很多盲点需要摸索.整合前面的系列文章,从Resource到BeanDefinition ...

  2. Spring源码学习笔记12——总结篇,IOC,Bean的生命周期,三大扩展点

    Spring源码学习笔记12--总结篇,IOC,Bean的生命周期,三大扩展点 参考了Spring 官网文档 https://docs.spring.io/spring-framework/docs/ ...

  3. 【Spring源码分析】Bean加载流程概览

    代码入口 之前写文章都会啰啰嗦嗦一大堆再开始,进入[Spring源码分析]这个板块就直接切入正题了. 很多朋友可能想看Spring源码,但是不知道应当如何入手去看,这个可以理解:Java开发者通常从事 ...

  4. 【Spring源码分析】非懒加载的单例Bean初始化前后的一些操作

    前言 之前两篇文章[Spring源码分析]非懒加载的单例Bean初始化过程(上篇)和[Spring源码分析]非懒加载的单例Bean初始化过程(下篇)比较详细地分析了非懒加载的单例Bean的初始化过程, ...

  5. 框架源码系列六:Spring源码学习之Spring IOC源码学习

    Spring 源码学习过程: 一.搞明白IOC能做什么,是怎么做的  1. 搞明白IOC能做什么? IOC是用为用户创建.管理实例对象的.用户需要实例对象时只需要向IOC容器获取就行了,不用自己去创建 ...

  6. spring源码分析系列 (5) spring BeanFactoryPostProcessor拓展类PropertyPlaceholderConfigurer、PropertySourcesPlaceholderConfigurer解析

    更多文章点击--spring源码分析系列 主要分析内容: 1.拓展类简述: 拓展类使用demo和自定义替换符号 2.继承图UML解析和源码分析 (源码基于spring 5.1.3.RELEASE分析) ...

  7. 【Spring源码分析】Bean加载流程概览(转)

    转载自:https://www.cnblogs.com/xrq730/p/6285358.html 代码入口 之前写文章都会啰啰嗦嗦一大堆再开始,进入[Spring源码分析]这个板块就直接切入正题了. ...

  8. Ioc容器beanDefinition-Spring 源码系列(1)

    Ioc容器beanDefinition-Spring 源码系列(1) 目录: Ioc容器beanDefinition-Spring 源码(1) Ioc容器依赖注入-Spring 源码(2) Ioc容器 ...

  9. Spring源码分析:非懒加载的单例Bean初始化前后的一些操作

    之前两篇文章Spring源码分析:非懒加载的单例Bean初始化过程(上)和Spring源码分析:非懒加载的单例Bean初始化过程(下)比较详细地分析了非懒加载的单例Bean的初始化过程,整个流程始于A ...

随机推荐

  1. 零基础入门Python游戏学习笔记(1)

    书是车洪于2020年出的,到手已经过去一年多了.现在学来,好多东西不一样了. 作者的GitHub,大家知道的原因,并不好打开. 代码就不搬了,只是为了学习方便,书籍勘误搬一下. 一.开发环境: 1.p ...

  2. java-23种设计模式概述【软件设计模式基本介绍(是什么、作用、优点)、模式的分类和介绍】

    一.设计模式基本介绍(是什么.作用.优点) 1.软件设计模式是什么? 软件设计模式(Software Design Pattern),又称设计模式. 2.设计模式的作用 ★ 提高代码的可复用性.可维护 ...

  3. PentestBox在win10里打不开工具 显示无系统命令的解决方法

    PentestBox详细安装过程:http://www.cnblogs.com/ESHLkangi/p/8336398.html 在使用PentestBox的时候出现了打不开工具的问题,最后看到一个大 ...

  4. HTTP攻击与防范-跨站攻击-01简介

    实验目的 1.掌握WEB渗透测试跨站攻击原理 2.了解WEB站点的跨站攻击脆弱性 3.修复存在跨站攻击可能的漏洞 实验原理 XSS又叫CSS (Cross Site script) ,跨站脚本攻击.它 ...

  5. 【基础知识】CPU 是如何工作的 |CPU 通过总线读取内存的工作方式

    一.简单cpu  是如何工作 方式讲解 CPU 的根本任务就是执行指令,对计算机来说最终都是一串由 0 和 1 组成的序列.CPU 从逻辑上可以划分成 3 个模块,分别是控制单元.运算单元和存储单元 ...

  6. linux基础-jdk1.8和weblogic12.2.1.3.0安装

    转至:https://www.cnblogs.com/jiarui-zjb/p/9642416.html 1.环境探查与准备 安装jdk和weblogic前需要对进行安装的linux系统硬件和软件环境 ...

  7. spring 中<ref parent="">标签是什么意思;ref标签与ref属性有什么不同;子容器如何引用父容器的bean

    spring的配置文件可能会有多个<property name="a" ref="b" />就是找当前配置文件里的bean 也就是id为b的 < ...

  8. jq给手机号加密

    效果: HTML代码:     <!-- 1手机绑定 -->     <div class="memberuser_box">         <di ...

  9. 鼠标点击的时候出现 "双心心" 的效果

    设置步骤 点击博客园的 [管理]  → [设置] → 一直往下拉, 找到 [页首Html代码],添加如下代码内容, 保存即可: <script type="text/javascrip ...

  10. linux基本篇--入门成神之路

    一.linux基础操作 1.初识bash shell概念  shell是系统的用户界面,提供了用户与内核进行交互操作的一种接口,他接受用户输入的命令并把它送入内核去执行,实际上shell是一个命令解释 ...