spring源码:BeanPostProcessor(li)
在spring管理Bean的初始化过程中,除了正常管理bean的实例化(初始化、参数注入等)外,还对外提供了丰富的对Bean操作的扩展。例如自定义初始化操作,自定义容器退出时Bean的销毁操作等等。这段时间看源码觉得此方面最实际的一个例子就是,我们在Bean初始化之前以及之后,框架允许我们做一些统一性的逻辑操作。BeanPostProcessor就完成了这个功能,它能够在装配Bean的过程中动态去改变Bean的行为,达到对Bean的增强。首先需要了解整个spring容器对bean初始化的流程,其流程图如下:

上图中已经完成了容器的初始化(对配置文件的加载和解析),已经进入到实例化阶段。能够看出BeanPostProcessor在整个流程中所处的位置,它正好围绕着bean自定义初始化方法前后。那么有一个问题,就是既然能够使用afterPropertySet或者init-method来自行初始化我们的bean,那么为什么还要这些BeanPostProcessor呢?因为afterPropertySet和init-method更关注自身的业务逻辑处理,而BeanPostProcessor更加通用,比如是否需要感知当前环境的上下文等更具有通用性的逻辑。
一、使用BeanPostProcessor来完成Bean实例化前、后处理
BeanPostProcessor接口定义了两个方法,postProcessBeforeInitialization和postProcessAfterInitialization。一看名字便一目了然它们各自的作用,那就是在初始化的前后进行预处理和后处理。问题的关键在于容器启动后,BeanFactory是怎么感知到这些BeanPostProcessor的存在呢?或者说我们自定义的BeanPostProcessor怎么能够集成到框架中,让容器感知和使用我们自定义的processor?首先,我们要找到BeanFactory对BeanPostProcessor的处理逻辑代码,这段代码在AbstractAutowireCapableBeanFactory类的initializeBean方法中:
Object wrappedBean = bean; //进入initializeBean方法之前,已经构造了一个Bean对象,不过未经初始化
if (mbd == null || !mbd.isSynthetic()) {
wrappedBean = applyBeanPostProcessorsBeforeInitialization(wrappedBean, beanName); //前处理
} try {
invokeInitMethods(beanName, wrappedBean, mbd);//这里有机会调用Bean的afterPropertySet方法以及自定义的初始化方法
}
catch (Throwable ex) {
throw new BeanCreationException(
(mbd != null ? mbd.getResourceDescription() : null),
beanName, "Invocation of init method failed", ex);
} if (mbd == null || !mbd.isSynthetic()) {
wrappedBean = applyBeanPostProcessorsAfterInitialization(wrappedBean, beanName);//后处理
}
return wrappedBean;
再进一步看applyBeanPostProcessorsBeforeInitialization方法。
Object result = existingBean;
// 在这里容器会遍历所有已经注册的BeanPostProcessor,无论是spring框架自身的还是我们自定义的。
for (Iterator it = getBeanPostProcessors().iterator(); it.hasNext();) {
BeanPostProcessor beanProcessor = (BeanPostProcessor) it.next();
result = beanProcessor.postProcessBeforeInitialization(result, beanName);
}
return result;
可以看到,我们完全有机会在afterPropertySet方法之前就对Bean进行操作,我们只要在BeanFactory中注册我们的BeanPostProcessor(在BeanFactory中注册我们的processor),那么就会在BeanFactory加载Bean的过程中,回调我们自己的BeanPostProcessor来完成我们的动作(上面代码中循环查找所有注册的BeanPostProcessor)。不过使用BeanPostProcessor带来的一个问题就是任何Bean的加载都会调用你注册过的这个processor,所以一些个性化的业务初始化操作放在这里是不合适的,业务初始化操作应该放到afterPropertySet(需要实现InitializingBean接口,侵入性较大)或者自定义初始化方法(需要在bean配置文件中指定init-method参数,侵入性较小)中完成。所以,BeanPostProcessor的使用应该是通用的业务逻辑处理。比如我们经常使用到的自动绑定(Autowired)就是使用BeanPostProcessor来完成的。我们来简单看一下处理autowired的这个processor是如何完成自动绑定成员变量和方法参数的。Autowired的processor类是:
AutowiredAnnotationBeanPostProcessor.class
autowired的过程都是在bean实例化之后做的,所以我们看postProcessAfterInitialization方法的具体实现。
// 根据当前bean对象的class信息得到使用注解autowired的成员变量和方法
InjectionMetadata metadata = findAutowiringMetadata(bean.getClass());
try {
/*
// 实现了将容器中的bean注入到当前对象成员变量的过程。这里之所以能够拿到容器中的其他bean,是因为AutowiredAnnotationBeanPostProcessor实现了BeanFactoryAware接口,从而得到当前容器的BeanFactory。另外具体看进去里面的实现,autowired的实现实现方式是通过byType实现的。 */
metadata.injectFields(bean, beanName);
}
catch (Throwable ex) {
throw new BeanCreationException(beanName, "Autowiring of fields failed", ex);
}
return true;
spring对bean进行初始化的过程默认使用set方法注入。而且属性注入是在初始化bean之前(参照AbstractAutowireCapableBeanFactory类的doCreateBean方法),也就是说我们的autowired方式应该是在set方法注入之后,autowired是会覆盖set方法注入的属性。我们可以做以下实验来验证我们的猜测:我们在Apple中准备注入另一个Bean seed,同时使用了set方法和autowired。Apple类中代码部分如下:
@Autowired
@Qualifier("seed1")
private Seed seed; public Sort getSeed() {
return seed;
} public void setSeed(Seed seed) {
this.seed = seed;
}
配置文件如下:
<bean id="apple" class="com.alibaba.china.Apple" >
<property name="seed">
<ref bean="seed0" />
</property>
</bean> <bean id="seed0" class="com.alibaba.china.Seed" >
<property name="seedName">
<value>苹果种子</value>
</property>
</bean> <bean id="seed1" class="com.alibaba.china.Seed">
<property name="seedName">
<value>Apple Seed</value>
</property>
</bean>
我们用set方法注入seed0,用自动绑定的方式注入seed1。由于BeanPostProcessor在初始化过程中执行,在set属性注入之后,所以预期结果seed1被注入。但是运行结果发现是seed0注入了apple对象。
为什么会这样呢?经过调试发现,在使用set方法属性注入的过程之前,会处理一类特殊的BeanPostProcessor------InstantiationAwareBeanPostProcessor,它是BeanPostProcessor的扩展接口,但是容器会在set注入之前优先检查是否存在这类BeanPostProcessor,如果存在InstantiationAwareBeanPostProcessor先执行这类processor。而恰恰巧合的是,AutowiredAnnotationBeanPostProcessor是InstantiationAwareBeanPostProcessor的子类。所以autowired的动作在set方法前做了,set方法的属性覆盖了autowired的属性,所以在上述例子中seed0最终被成功注入到apple这个对象里面。我们用类图能更加形象地描述InstantiationAwareBeanPostProcessor与BeanPostProcessor接口:

当获取一个bean对象时,主要逻辑在AbstractAutowireCapableBeanFactory类的doCreateBean方法中。其父类中维护一个beanPostProcessors列表,我们可以在容器启动时加入我们的processor,在执行到doCreateBean中我们的processor会被回调。我们根据具体的需要去实现BeanPostProcessor接口或者InstantiationAwareBeanPostProcessor接口。前者主要是初始化前后的处理动作,而后者主要是执行实例化前后的逻辑。
ApplicationContext更加简单地使用BeanPostProcessor:
如果我们直接使用BeanFactory,需要把我们的BeanPostProcessor注册到BeanFactory上,否则BeanFactory感知不到我们的processor。而ApplicationContext则在此方面做了增强,我们只需在配置文件里面声明我们的BeanPostProcessor,ApplicationContext能够自动做到感知。ApplicationContext是如何做到的呢?我们不妨自己先猜想一下,如果让我们自己实现这个增强功能应该怎么做?大致方向就是自己去扫描配置文件,看所有的bean中有哪些是BeanPostProcessor,然后我们不分青红皂白一股脑把所有的BeanPostProcessor都注册进去。spring对ApplicationContext也正是这么实现的,具体的描述在另一片分享中更具体的提到:
二、
spring源码:BeanPostProcessor(li)的更多相关文章
- 2.spring源码-BeanPostProcessor后置处理之ApplicationContextAwareProcessor,实现spring容器中某一个类的bean对象在初始化时需要得到Spring容器内容。
需求:我们的需求是,在spring初始化完毕时,使我们自定义一个类Bird类可以得到spring容器内容. 实现步骤: 1.首先我们来看一下ApplicationContextAwareProcess ...
- 1.spring源码-BeanPostProcessor后置处理器
1.BeanPostProcessor接口的介绍: BeanPostProcessor是一个接口,其中有两个方法,postProcessBeforeInitialization和postProcess ...
- spring源码:学习线索(li)
一.spring xml配置(不包括AOP,主要了解在初始化及实例化过程中spring配置文件中每项内容的具体实现过程,从根本上掌握spring) <bean>的名字 &,alia ...
- [spring源码学习]五-BeanPostProcessor的使用
一.接口描述 spring提供了一个接口类-BeanPostProcessor,我们叫他:bean的加工器,应该是在bean的实例化过程中对bean做一些包装处理,里边提供两个方法 public in ...
- Ioc容器BeanPostProcessor-Spring 源码系列(3)
Ioc容器BeanPostProcessor-Spring 源码系列(3) 目录: Ioc容器beanDefinition-Spring 源码(1) Ioc容器依赖注入-Spring 源码(2) Io ...
- spring源码分析系列 (2) spring拓展接口BeanPostProcessor
Spring更多分析--spring源码分析系列 主要分析内容: 一.BeanPostProcessor简述与demo示例 二.BeanPostProcessor源码分析:注册时机和触发点 (源码基于 ...
- Spring 源码(8)Spring BeanPostProcessor的注册、国际化及事件发布机制
上一篇文章https://www.cnblogs.com/redwinter/p/16198942.html介绍了Spring的注解的解析过程以及Spring Boot自动装配的原理,大概回顾下:Sp ...
- spring源码:web容器启动(li)
web项目中可以集成spring的ApplicationContext进行bean的管理,这样使用起来bean更加便捷,能够利用到很多spring的特性.我们比较常用的web容器有jetty,tomc ...
- spring源码分析(一)IoC、DI
创建日期:2016.08.06 修改日期:2016.08.07 - 2016.08.12 交流QQ:992591601 参考书籍:<spring源码深度解析>.<spring技术内幕 ...
- spring源码分析系列 (5) spring BeanFactoryPostProcessor拓展类PropertyPlaceholderConfigurer、PropertySourcesPlaceholderConfigurer解析
更多文章点击--spring源码分析系列 主要分析内容: 1.拓展类简述: 拓展类使用demo和自定义替换符号 2.继承图UML解析和源码分析 (源码基于spring 5.1.3.RELEASE分析) ...
随机推荐
- bzoj1901--树状数组套主席树
树状数组套主席树模板题... 题目大意: 给定一个含有n个数的序列a[1],a[2],a[3]--a[n],程序必须回答这样的询问:对于给定的i,j,k,在a[i],a[i+1],a[i+2]--a[ ...
- python 3.5 成功安装 scrapy 的步骤
http://www.cnblogs.com/hhh5460/p/5814275.html
- 海康网络摄像机YV12转换为BGR,由opencv Mat显示 (转)
我使用的是海康DS-2CD852MF-E, 200万,网络摄像机,已经比较老了,不过SDK在海康官网下载的,开发流程都差不多. 海康摄像机回调解码后的视频数据格式为YV12,顺便说一下YV12的数据格 ...
- PHP设计模式(三)抽象工厂模式(Abstract Factory For PHP)
一.什么是抽象工厂模式 抽象工厂模式的用意为:给客户端提供一个接口,可以创建多个产品族中的产品对象 ,而且使用抽象工厂模式还要满足以下条件: 系统中有多个产品族,而系统一次只可能消费其中一族产品. 同 ...
- JavaScript学习笔记(一)——延迟对象、跨域、模板引擎、弹出层、AJAX示例
一.AJAX示例 AJAX全称为“Asynchronous JavaScript And XML”(异步JavaScript和XML) 是指一种创建交互式网页应用的开发技术.改善用户体验,实现无刷新效 ...
- BPM配置故事之案例14-数据字典与数据联动
小明遇到了点麻烦,他昨天又收到了行政主管发来的邮件,要求把出差申请单改由H3 BPM进行,表单如下 行政主管的出差申请表 小明对表单进行了调整,设计出了一份适合在系统中使用的表单,但在"出差 ...
- Android—关于自定义对话框的工具类
开发中有很多地方会用到自定义对话框,为了避免不必要的城府代码,在此总结出一个工具类. 弹出对话框的地方很多,但是都大同小异,不同无非就是提示内容或者图片不同,下面这个类是将提示内容和图片放到了自定义函 ...
- 萌新笔记——linux下查看内存的使用情况
windows上有各种软件可以进行"一键加速"之类的操作,释放掉一些内存(虽然我暂时不知道是怎么办到的,有待后续学习).而任务管理器也可以很方便地查看各进程使用的内存情况,如下图: ...
- 项目游戏开发日记 No.0x00000
14软二杨近星(2014551622) ---恢复内容开始--- 2016-03-17 从开始迈进软件工程专业, 已经快两年了, 记得当初选择软件的理由是, 我要学去做东西, 我享受开发过程. 两年来 ...
- 四、可空类型Nullable<T>到底是什么鬼
值类型为什么不可以为空 首先我们都知道引用类型默认值都是null,而值类型的默认值都有非null. 为什么引用类型可以为空?因为引用类型变量都是保存一个对象的地址引用(就像一个url对应一个页面),而 ...