前言
结束了前面的基础结构分析,了解到 Spring
是如何识别配置文件和进行解析属性,最终将 bean
加载到内存中。同时为了更好得理解 Spring
的扩展功能,我们先来巩固一下 beanFactory
和 bean
的概念,然后再分析新内容后处理器 PostProcessor
。
首先我们先将 Spring
想像成一个大容器,然后保存了很多 bean
的信息,根据定义:bean
是一个被实例化,组装,并通过 Spring IoC
容器所管理的对象,也可以简单得理解为我们在配置文件配置好元数据,Spring IoC
容器会帮我们对 bean
进行管理,这些对象在=使用的时候通过 Spring
取出就能使用。
那么是谁帮这个 Spring
管理呢,那就是 BeanFactory
,粗暴点直译为 bean
工厂,但其实它才是承担容器功能的幕后实现者,它是一个接口,提供了获取 bean
、获取别名 Alias
、判断单例、类型是否匹配、是否原型等方法定义,所以需要通过引用,实现具体方法才后才能使用。
回顾完 beanFactory
后,我们再来回顾在前面内容中,看到过很多后处理器 PostProcessor
的代码影子,分别是 BeanFactoryPostProcessor
:主体是 BeanFactory
, 和 BeanPostProcessor
:主体是 Bean
,这两者都是 Spring
用来为使用者提供的扩展功能之一。
接下来为了更好的分析和了解使用后处理器,实现扩展功能,一起跟踪源码学习吧~
BeanFactoryPostProcessor
是什么
BeanFactoryPostProcessor
是一个接口,在里面只有一个方法定义:
1 |
|
定义:是 Spring
对外提供可扩展的接口,能够在容器加载了所有 bean
信息(AbstractApplicationContext#obtainFreshBeanFactory
方法)之后,bean
实例化之前执行,用来修改 bean
的定义属性。
可以看到,方法参数是 ConfigurableListableBeanFactory beanFactory
,说明我们可以通过引用该接口,在方法中实现逻辑,对容器中 bean
的定义(配置元数据)进行处理
同时,执行后处理器是有先后顺序的概念,我们可以通过设置 order
属性来控制它们的执行次序,前提是 BeanFactoryPostProcessor
实现了 Order
接口。
下面一起来看下它的如何使用,以及是如何进行注册和执行~
如何使用
官方例子:PropertyPlaceholderConfigurer
这个类是 Spring
容器里自带的后处理器,是用来替换占位符,填充属性到 bean
中。
像我们在 xml
文件中配置了属性值为 ${max.threads}
,能够通过它来找到 max.threads
在配置文件对应的值,然后将属性填充到 bean
中。
虽然在 Spring 5 中,PropertyPlaceholderConfigurer
已经打上了不建议使用的标志 @Deprecated
,看了文件注释,提示我们去使用 Environment
来设置属性,但我觉得这个后处理器的思想是一样的,所以还是拿它作为例子进行熟悉。
先来看下它的继承体系:
忽略它被冷落的下划线标签
当 Spring
加载任何实现了 BeanFactoryPostProcessor
接口的 bean
配置时,都会在 bean
工厂载入所有 bean
的配置之后执行 postProcessBeanFactory
方法。
可以看到它引用了 BeanFactoryPostProcessor
接口,在 PropertyResourceConfigurer
父类中实现了 postProcessBeanFactory
,在方法中依次调用了合并资源 mergedProps
方法,属性转换 convertProperties
方法和真正修改 beanFactory
中配置元数据的 processProperties(beanFactory, mergedProps)
方法。
因为通过在 PropertyPlaceholderConfigurer
的后处理方法 postProcessBeanFactory
,BeanFactory
在实例化任何 bean
之前获得配置信息,从而能够正确解析 bean
描述文件中的变量引用。
所以通过后处理器,我们能够对 beanFactory
中的 bean
配置信息在实例化前还有机会进行修改。
题外话:想到之前我遇到全半角空格的配置问题,程序认为全半角空格不是同一个字符,但肉眼却很难察觉,所以感觉可以在加载配置信息时,通过自定义一个后处理,在实例化之前,将全角空格转成半角空格,这样程序比较时都变成统一样式。所以后处理器提供的扩展功能可以让我们对想要处理的 bean
配置信息进行特定修改
使用自定义 BeanFactoryPostProcessor
实现的功能与书中的类似,例如之前西安奔驰汽车维权事件,如果相关网站想要屏蔽这奔驰这两个字,可以通过后处理器进行替换:
1. 配置文件 factory-post-processor.xml
1 | <bean id="carPostProcessor" class="context.CarBeanFactoryPostProcessor"> |
2. 后处理器 CarBeanFactoryPostProcessor
1 | public class CarBeanFactoryPostProcessor implements BeanFactoryPostProcessor { |
3. 启动
1 | public class BeanFactoryPostProcessorBootstrap { |
通过上面的演示代码,新增一个自定义实现 BeanFactoryPostProcessor
的后处理器 CarBeanFactoryPostProcessor
,在 postProcessBeanFactory
方法中进行逻辑处理,最后通过 visitor.visitBeanDefinition
修改配置信息。
查看输出结果,能发现宝马敏感词已经被屏蔽了,实现了后处理器的逻辑功能~
在哪注册
按照一般套路,后处理器需要有个地方进行注册,然后才能进行执行,通过代码分析,的确在 AbstractApplicationContext
中看到了 beanFactoryPostProcessors
数组列表,但往数组中添加后处理器的方法 addBeanFactoryPostProcessor
只在单元测试包调用了。
这让我很迷惑它到底是在哪里进行注册,直到我看到它的执行方法,原来我们定义的后处理器在 bean
信息加载时就放入注册表中,然后通过 beanFactory.getBeanNamesForType(BeanDefinitionRegistryPostProcessor.class, true, false)
方法获取后处理器列表遍历执行。
所以前面的 beanFactoryPostProcessors
数组列表,是让我们通过硬编码方法方式,手动添加进去,然后通过 context.refresh()
方法后,再执行硬编码的后处理器
例如下面这个例子
1 | public class HardCodeBeanFactoryPostProcessor implements BeanFactoryPostProcessor { |
激活 BeanFactoryPostProcessor
看完了怎么使用后,我们来分析下 Spring
是如何识别和执行 BeanFactoryPostProcessor
,入口方法:
org.springframework.context.support.AbstractApplicationContext#invokeBeanFactoryPostProcessors
实际上,委派了 PostProcessorRegistrationDelegate.invokeBeanFactoryPostProcessors(beanFactory, getBeanFactoryPostProcessors())
代理进行执行。由于代码有点长,所以我画了一个流程图,可以结合流程图来分析代码:
流程图
代码
1 | public static void invokeBeanFactoryPostProcessors(ConfigurableListableBeanFactory beanFactory, List<BeanFactoryPostProcessor> beanFactoryPostProcessors) { |
从上面贴的代码中能够看到,对于 beanFactoryPostProcessor
的处理主要分两种情况:
beanFactory
是BeanDefinitionRegistry
类型:需要特殊处理beanFactory
不是BeanDefinitionRegistry
类型:进行普通处理
对于每种情况都需要考虑硬编码注入注册的后处理器(上面已经提到如何进行硬编码)以及通过配置注入的后处理器
对于 BeanDefinitionRegistry
类型的处理器的处理主要包括以下内容:
- 处理硬编码注册的后处理器
- 记录后处理器主要使用以下三个
List
registryPostProcessors
:记录通过硬编码方式注册的BeanDefinitionRegistryPostProcessor
regularPostProcessors
:记录通过硬编码方式注册的BeanFactoryPostProcessor
regitstryPostProcessorBeans
:记录通过配置方式注册的BeanDefinitionRegistryPostProcessor
- 对于以上后处理器列表,统一调用
BeanFactoryPostProcessor
的postProcessBeanFactory
方法 - 对
beanFactoryPostProcessors
中非BeanDefinitionRegistryPostProcessor
类型的后处理器进行统一的postProcessBeanFactory
方法 - 普通
beanFactory
处理:其实在这一步中,就是忽略了BeanDefinitionRegistryPostProcessor
类型,对BeanFactoryPostProcessor
进行直接处理。
流程图中描述了整体调用链路,具体调用方法在代码中的注释也描述出来了,所以结合起来看应该能够理解整体流程~
总结
本次分析了 beanFactory
的后处理器 BeanFactoryPostProcessor
,了解了 Spring
给我们提供的这个扩展接口使用用途和在源码中如何进行激活执行。
由于个人技术有限,如果有理解不到位或者错误的地方,请留下评论,我会根据朋友们的建议进行修正
因为我也是一边看书,一边做笔记,下载了源码。
代码和注释都在里面,小伙伴们可以下载我上传的代码,亲测可运行~
spring-analysis-note 码云 Gitee 地址
spring-analysis-note Github 地址
传送门:
参考资料
- Bean 定义
- Sping 的 BeanFactory 容器
- Spring拓展接口之BeanFactoryPostProcessor,占位符与敏感信息解密原理
- Spring 源码深度解析 / 郝佳编著. – 北京 : 人民邮电出版社