水稻:这两天看了BeanDefinition和BeanFactoryPostProcessor还有BeanPostProcessor的源码。要不要了解一下

菜瓜:six six six,大佬请讲

水稻:上次我们说SpringIOC容器是一个典型的工厂模式

  • 假如我们把Spring比作一个生产模型的大工厂,那么.class文件就是原材料。而BeanDefinition就是创建模型的模具。不管是传统的XML还是后面的注解,Spring在启动的时候都会创建一个扫描器去扫描指定目录下的.class文件,并根据文件的注解,实现的接口以及成员变量将其封装一个个的BeanDefinition。

    • 比较重要的属性有id,class,构造函数封装类,属性封装类,factoryMethod等
  • 在对象初始化之前Spring会完成BeanDefinition对象的解析并将其装入List容器beanDefinitionNames中,然后开始遍历该容器并根据BeanDefinition创建对象

菜瓜:sodasinei,BeanDefinition我了解了。它是创建bean的模板,类似于java创建对象依赖的class一样。那还有两个很长的单词是啥呢?

水稻:忽略掉后面老长的后缀,我们看BeanFactory和Bean是不是很亲切。PostProcessor被翻译成后置处理器,暂且我们把它看成是处理器就行

  • BeanFactory是bean工厂,它可以获取并修改BeanDefinition的属性,进而影响后面创建的对象。
  • Bean就是Spring的对象,这些个处理器才是真正处理bean对象的各个环节的工序,包括属性,注解,方法

菜瓜:有了模糊的概念,不明觉厉

水稻:来,看demo

package com.vip.qc.postprocessor;

import org.springframework.beans.BeansException;
import org.springframework.beans.MutablePropertyValues;
import org.springframework.beans.factory.config.BeanDefinition;
import org.springframework.beans.factory.config.BeanFactoryPostProcessor;
import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;
import org.springframework.stereotype.Component; /**
* 获取初始化好的BeanFactory,此时还未进行bean的实例化
*
* @author QuCheng on 2020/6/14.
*/
@Component
public class BeanFactoryPostProcessorT implements BeanFactoryPostProcessor { public static final String BEAN_NAME = "processorT"; @Override
public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
BeanDefinition initializingBeanT = beanFactory.getBeanDefinition(BEAN_NAME);
MutablePropertyValues propertyValues = initializingBeanT.getPropertyValues();
String pName = "a";
System.out.println("BeanFactoryPostProcessor a " + propertyValues.getPropertyValue(pName) + " -> 1");
propertyValues.addPropertyValue(pName, "1");
}
} package com.vip.qc.postprocessor; import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanPostProcessor;
import org.springframework.stereotype.Component; /**
* @author QuCheng on 2020/6/14.
*/
@Component
public class BeanPostProcessorT implements BeanPostProcessor { public static final String beanNameT = "processorT"; @Override
public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
if (beanNameT.equals(beanName)) {
ProcessorT processorT = ((ProcessorT) bean);
System.out.println("BeanPostProcessor BeforeInitialization a:" + processorT.getA() + "-> 3");
processorT.setA("3");
}
return bean;
} @Override
public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
if (beanNameT.equals(beanName)){
ProcessorT processorT = ((ProcessorT) bean);
System.out.println("BeanPostProcessor AfterInitialization a:" + processorT.getA() + "-> 4");
processorT.setA("4");
}
return bean;
} } package com.vip.qc.postprocessor; import org.springframework.stereotype.Component; /**
* @author QuCheng on 2020/6/14.
*/
@Component
public class ProcessorT { public ProcessorT() {
System.out.println("ProcessorT 无参构造 a:" + a + "-> 2" );
a = "2";
} private String a; public String getA() {
return a;
} public void setA(String a) {
this.a = a;
} @Override
public String toString() {
return "ProcessorT{" +
"a='" + a + '\'' +
'}';
}
} // 测试类
@Test
public void test() {
AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext("com.vip.qc.postprocessor");
ProcessorT processorT = (ProcessorT) context.getBean("processorT");
System.out.println(processorT);
} // 结果
BeanFactoryPostProcessor a null -> 1
ProcessorT 无参构造 a:null-> 2
BeanPostProcessor BeforeInitialization a:1-> 3
BeanPostProcessor AfterInitialization a:3-> 4
ProcessorT{a='4'}
  • BeanFactoryPostProcessor在对象还未初始化前可以拿到对象的BeanDefinition对其设置属性值  

  • 过程中我们分别对属性a设置了1,2,3,4的值。最后我们拿到的值为4

菜瓜:好像看懂了。BeanFactoryPostProcessor可以拿到BeanFactory对象,获取里面所有的BeanDefinition并可对其进行干预。BeanPostProcessor其实是在bean已经被创建完成之后进行加工操作

水稻:没错。这是我们自己进行干预的demo。限于篇幅有限,你可以去看一下Spring自己对于这两个接口的实现源码。比较重要的推荐下面几个

  • ConfigurationClassPostProcessor 实现BeanFactoryPostProcessor子接口

    • 完成对@Configuration、@Component、@ComponentScan、@Bean、@Import、@ImportSource注解的搜集和解析
    • @Bean注解会被封装成所在Bean的BeanDefinition中的factoryMethod属性中,单独进行实例化
  • CommonAnnotationBeanPostProcessor 实现 BeanPostProcessor
    • 完成@PostConstruct@PreDestroy@Resource注解的搜集和解析工作
    • @PostConstruct会在对象初始化且属性渲染完成后进行
    • @Resource注解(参照下面)
  • AutowiredAnnotationBeanPostProcessor 实现 BeanPostProcessor
    • 完成@Autowired@Value注解的搜集和解析工作
    • 在对象初始化完成之后会先进行注解的搜集,然后进行属性渲染调用populateBean方法,使用策略模式调用实现接口对注解进行解析,有@Autowired和@Value注解会调用getBean方法发起对依赖属性的注入
  • AbstractAutoProxyCreator的入口类也是实现的BeanPostProcessor

菜瓜:你放心,我不会看的。这么复杂的东西,听着都费劲

水稻:不愧是你!没事,有机会聊bean的生命周期的时候咱们还会说到这些东西。到时候再刷一遍

Spring:BeanDefinition&PostProcessor不了解一下吗?的更多相关文章

  1. Spring BeanDefinition的加载

     前面提到AbstractRefreshableApplicationContext在刷新BeanFactory时,会调用loadBeanDefinitions方法以加载系统中Bean的定义,下面将讲 ...

  2. 扯淡 Spring BeanDefinition

    相关文章 Spring 整体架构 编译Spring5.2.0源码 Spring-资源加载 Spring 容器的初始化 Spring-AliasRegistry Spring 获取单例流程(一) Spr ...

  3. Spring beanDefinition载入

    @Override public void refresh() throws BeansException, IllegalStateException { synchronized (this.st ...

  4. Spring BeanDefinition

    定义 /** * A BeanDefinition describes a bean instance, which has property values, * constructor argume ...

  5. spring BeanDefinition 继承结构图

    ConfigurationClassBeanDefinition 是ConfigurationClassBeanDefinitionReader的静态内部类

  6. Spring工厂方式创建Bean实例

    创建Bean实例的方式: 1) 通过构造器(有参或无参) 方式: <bean id="" class=""/> 2) 通过静态工厂方法 方式: &l ...

  7. Spring之一:IoC容器体系结构

    温故而知心. Spring IoC概述 常说spring的控制反转(依赖反转),看看维基百科的解释: 如果合作对象的引用或依赖关系的管理要由具体对象来完成,会导致代码的高度耦合和可测试性降低,这对复杂 ...

  8. Spring框架之beans源码完全解析

    导读:Spring可以说是Java企业开发里最重要的技术.而Spring两大核心IOC(Inversion of Control控制反转)和AOP(Aspect Oriented Programmin ...

  9. 【Spring】IoC容器 - Spring Bean作用域Scope(含SpringCloud中的RefreshScope )

    前言 上一章学习了[依赖来源],本章主要讨论SpringBean的作用域,我们这里讨论的Bean的作用域,很大程度都是默认只讨论依赖来源为[Spring BeanDefinition]的作用域,因为在 ...

随机推荐

  1. CF948D Perfect Security

    题目链接:http://codeforces.com/contest/948/problem/D 知识点: Trie 题目大意: 给出两个长度为 \(N(1 \le N \le 300000)\) 的 ...

  2. C# 数据操作系列 - 14 深入探索SqlSugar

    0.前言 在上一篇中,我们知道了如何使用SqlSugar,但是也只是简单的了解了如何使用,仿佛是套着镣铐行走,这明显不符合一个合格的程序员应有的素养.所以,这一篇我们将对其进行深挖,探究其背后的秘密. ...

  3. Text Reverse(hdu1062)

    输入方式:先输入整数,再循环输入字符串. 思考:字符串中有空格.那么要在字符串大循环输入前,首先,用"getchar()"函数读取scanf_s()函数缓冲区的空格或者空行或者换行 ...

  4. eclipse——管理远程资源的缓存,例如从Internet下载的资源。

    原文:Manage the cache of remote resources,such as those downloaded from the internet.

  5. js数组对象的一些常用方法

    pop:删除数组最后一个元素 语法: array.pop(); 如 var array = ['1','2','3']; array.pop(); 返回结果:[‘1’,‘2’]此方法会改变数组的长度 ...

  6. Kubernetes基本概念与架构

    Kubernetes,面向云原生应用的新“云平台” Kubernetes:以google Brog为原型 Kubernetes的成长历程: l  2014年,Kubernetes正式由google开源 ...

  7. 什么是cookie?

    cookie是什么? 其实cookies是由网络服务器存储在你电脑硬盘上的一个txt类型的小文件,它和你的网络浏览行为有关,所以存储在你电脑上的cookies就好像你的一张身份证,你电脑上的cooki ...

  8. 2.Redis安装和简单使用

    (1)安装Redis Redis目前只支持Linux系统,因为开发此软件的创始者认为,Redis是为后台数据服务的,所以认为该软件使用在纯净的服务环境下,而不是应用型操作系统下,而Linux作为服务器 ...

  9. [Node.js]001.安装与环境配置

    安装与环境配置 第一步:下载安装文件 第二步:安装nodejs 第三步:npm安装 第四步:安装相关环境 第五步:安装CoffeeScript 第六步:CoffeeScript测试实例 第一步:下载安 ...

  10. [JavaWeb基础] 018.Struts2 Action通配符使用

    Struts2中有一个很牛逼的action通配符,可以用来简化action配置,以我们将要讲解的案例来说,如果我们要对一个学生信息进行增加,删除,修改,那么按照原来的做法,我们需要写3个Action来 ...