Java充电社
专辑
博文
联系我
本人继续续收门徒,亲手指导
Spring系列第29篇:BeanFactory扩展(BeanFactoryPostProcessor、BeanDefinitionRegistryPostProcessor)
相关专辑:
Spring教程
<div style="display:none"></div> **Spring中有2个非常重要的接口:BeanFactoryPostProcessor和BeanDefinitionRegistryPostProcessor,这2个接口面试中也会经常问到,本文我们一起来拿下他们俩。** ## 先来看几个问题 1. BeanFactoryPostProcessor是做什么的? 2. BeanDefinitionRegistryPostProcessor是干什么的? 3. BeanFactoryPostProcessor和BeanDefinitionRegistryPostProcessor有什么区别? 4. 这几个接口的执行顺序是什么样的? ## Spring容器中主要的4个阶段 - 阶段1:Bean注册阶段,此阶段会完成所有bean的注册 - 阶段2:BeanFactory后置处理阶段 - 阶段3:注册BeanPostProcessor - 阶段4:bean创建阶段,此阶段完成所有单例bean的注册和装载操作,这个阶段不是我们本文关注的重点,有兴趣的,可以去看之前的文章中有详细介绍:<a href="/course/5/105" target="_blank">Bean生命周期详解</a> 本文介绍的2个接口主要和前2个阶段有关系,下面我们主要来看前2个阶段。 ## 阶段1:Bean注册阶段 ### 概述 spring中所有bean的注册都会在此阶段完成,按照规范,所有bean的注册必须在此阶段进行,其他阶段不要再进行bean的注册。 这个阶段spring为我们提供1个接口:BeanDefinitionRegistryPostProcessor,spring容器在这个阶段中会获取容器中所有类型为`BeanDefinitionRegistryPostProcessor`的bean,然后会调用他们的`postProcessBeanDefinitionRegistry`方法,源码如下,方法参数类型是`BeanDefinitionRegistry`,这个类型大家都比较熟悉,即bean定义注册器,内部提供了一些方法可以用来向容器中注册bean。 ```java public interface BeanDefinitionRegistryPostProcessor extends BeanFactoryPostProcessor { void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException; } ``` 这个接口还继承了`BeanFactoryPostProcessor`接口,这个大家先不用关心,一会阶段2中会介绍。 当容器中有多个`BeanDefinitionRegistryPostProcessor`的时候,可以通过下面任意一种方式来指定顺序 1. 实现`org.springframework.core.PriorityOrdered`接口 2. 实现`org.springframework.core.Ordered`接口 执行顺序: ```sql PriorityOrdered.getOrder() asc,Ordered.getOrder() asc ``` 下面通过案例来感受一下效果。 ### 案例1:简单实用 此案例演示`BeanDefinitionRegistryPostProcessor`的简单使用 #### 自定义一个BeanDefinitionRegistryPostProcessor 下面我们定义了一个类,需要实现`BeanDefinitionRegistryPostProcessor`接口,然后会让我们实现2个方法,大家重点关注`postProcessBeanDefinitionRegistry`这个方法,另外一个方法来自于`BeanFactoryPostProcessor`,一会我们后面在介绍这个方法,在`postProcessBeanDefinitionRegistry`方法中,我们定义了一个bean,然后通过`registry`将其注册到容器了,代码很简单 ```java package com.javacode2018.lesson003.demo3.test0; @Component public class MyBeanDefinitionRegistryPostProcessor implements BeanDefinitionRegistryPostProcessor { @Override public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException { //定义一个字符串类型的bean AbstractBeanDefinition userNameBdf = BeanDefinitionBuilder. genericBeanDefinition(String.class). addConstructorArgValue("路人"). getBeanDefinition(); //将userNameBdf注册到spring容器中 registry.registerBeanDefinition("userName", userNameBdf); } @Override public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException { } } ``` #### 同包中来个配置类 ```java package com.javacode2018.lesson003.demo3.test0; import org.springframework.context.annotation.ComponentScan; @ComponentScan public class MainConfig0 { } ``` #### 测试用例 ```java package com.javacode2018.lesson003.demo3; import com.javacode2018.lesson003.demo3.test0.MainConfig0; import com.javacode2018.lesson003.demo3.test1.MainConfig1; import org.junit.Test; import org.springframework.context.annotation.AnnotationConfigApplicationContext; public class BeanDefinitionRegistryPostProcessorTest { @Test public void test0() { AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(); context.register(MainConfig0.class); context.refresh(); System.out.println(context.getBean("userName")); } } ``` #### 运行输出 ```java 路人 ``` ### 案例2:多个指定顺序 下面我们定义2个`BeanDefinitionRegistryPostProcessor`,都实现`Ordered`接口,第一个order的值为2,第二个order的值为1,我们来看一下具体执行的顺序。 #### 第一个 ```java package com.javacode2018.lesson003.demo3.test1; @Component public class BeanDefinitionRegistryPostProcessor1 implements BeanDefinitionRegistryPostProcessor, Ordered { @Override public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException { System.out.println(String.format("BeanDefinitionRegistryPostProcessor1{order=%d},注册name bean,", this.getOrder())); //定义一个bean AbstractBeanDefinition nameBdf = BeanDefinitionBuilder. genericBeanDefinition(String.class). addConstructorArgValue("路人甲java"). getBeanDefinition(); //将定义的bean注册到容器 registry.registerBeanDefinition("name", nameBdf); } @Override public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException { } @Override public int getOrder() { return 2; } } ``` #### 第二个 ```java package com.javacode2018.lesson003.demo3.test1; @Component public class BeanDefinitionRegistryPostProcessor2 implements BeanDefinitionRegistryPostProcessor, Ordered { @Override public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException { System.out.println(String.format("BeanDefinitionRegistryPostProcessor2{order=%d},注册car bean,", this.getOrder())); //定义一个bean AbstractBeanDefinition nameBdf = BeanDefinitionBuilder. genericBeanDefinition(String.class). addConstructorArgValue("保时捷"). getBeanDefinition(); //将定义的bean注册到容器 registry.registerBeanDefinition("car", nameBdf); } @Override public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException { } @Override public int getOrder() { return 1; } } ``` > 上面2个类中的postProcessBeanDefinitionRegistry方法第一行都有输出,一个可以通过运行结果看到执行的顺序。 #### 同包中添加配置类 ```java package com.javacode2018.lesson003.demo3.test1; import org.springframework.context.annotation.ComponentScan; @ComponentScan public class MainConfig1 { } ``` #### 测试案例 ```java @Test public void test1() { AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(); context.register(MainConfig1.class); context.refresh(); context.getBeansOfType(String.class).forEach((beanName, bean) -> { System.out.println(String.format("%s->%s", beanName, bean)); }); } ``` #### 运行输出 ```java BeanDefinitionRegistryPostProcessor2{order=1},注册car bean, BeanDefinitionRegistryPostProcessor1{order=2},注册name bean, car->保时捷 name->路人甲java ``` ### 小结 `BeanDefinitionRegistryPostProcessor`有个非常重要的实现类: ```java org.springframework.context.annotation.ConfigurationClassPostProcessor ``` 这个类可能有些人不熟悉,下面这些注解大家应该比较熟悉吧,这些注解都是在上面这个类中实现的,通过这些注解来实现bean的批量注册 ```java @Configuration @ComponentScan @Import @ImportResource @PropertySource ``` 有兴趣的朋友可以去看一下`ConfigurationClassPostProcessor#postProcessBeanDefinitionRegistry`研究一下上面这些注解的解析过程,可以学到很多东西。 ## 阶段2:BeanFactory后置处理阶段 ### 概述 到这个阶段的时候,spring容器已经完成了所有bean的注册,这个阶段中你可以对BeanFactory中的一些信息进行修改,比如修改阶段1中一些bean的定义信息,修改BeanFactory的一些配置等等,此阶段spring也提供了一个接口来进行扩展:`BeanFactoryPostProcessor`,简称`bfpp`,接口中有个方法`postProcessBeanFactory`,spring会获取容器中所有BeanFactoryPostProcessor类型的bean,然后调用他们的`postProcessBeanFactory`,来看一下这个接口的源码: ```java @FunctionalInterface public interface BeanFactoryPostProcessor { void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException; } ``` 当容器中有多个`BeanFactoryPostProcessor`的时候,可以通过下面任意一种方式来指定顺序 1. 实现`org.springframework.core.PriorityOrdered`接口 2. 实现`org.springframework.core.Ordered`接口 执行顺序: ```sql PriorityOrdered.getOrder() asc,Ordered.getOrder() asc ``` 下面通过案例来感受一下效果。 ### 案例 这个案例中演示,在BeanFactoryPostProcessor来修改bean中已经注册的bean定义的信息,给一个bean属性设置一个值。 #### 先来定义一个bean ```java package com.javacode2018.lesson003.demo3.test2; import org.springframework.stereotype.Component; @Component public class LessonModel { //课程名称 private String name; public String getName() { return name; } public void setName(String name) { this.name = name; } @Override public String toString() { return "LessonModel{" + "name='" + name + '\'' + '}'; } } ``` 上面这个bean有个name字段,并没有设置值,下面我们在BeanFactoryPostProcessor来对其设置值。 #### 自定义的BeanFactoryPostProcessor 下面代码中,我们先获取`lessonModel`这个bean的定义信息,然后给其`name`属性设置了一个值。 ```java package com.javacode2018.lesson003.demo3.test2; import org.springframework.beans.BeansException; 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; @Component public class MyBeanFactoryPostProcessor implements BeanFactoryPostProcessor { @Override public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException { System.out.println("准备修改lessonModel bean定义信息!"); BeanDefinition beanDefinition = beanFactory.getBeanDefinition("lessonModel"); beanDefinition.getPropertyValues().add("name", "spring高手系列!"); } } ``` #### 测试用例 ```java @Test public void test2() { AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(); context.register(MainConfig2.class); context.refresh(); System.out.println(context.getBean(LessonModel.class)); } ``` #### 运行输出 ```java 准备修改lessonModel bean定义信息! LessonModel{name='spring高手系列!'} ``` 结果中可以看出,通过`BeanFactoryPostProcessor`修改了容器中已经注册的bean定义信息。 ### 这个接口的几个重要实现类 #### PropertySourcesPlaceholderConfigurer 这个接口做什么的,大家知道么?来看一段代码 ```java <bean class="xxxxx"> <property name="userName" value="${userName}"/> <property name="address" value="${address}"/> </bean> ``` 这个大家比较熟悉吧,spring就是在`PropertySourcesPlaceholderConfigurer#postProcessBeanFactory`中来处理xml中属性中的`${xxx}`,会对这种格式的进行解析处理为真正的值。 #### CustomScopeConfigurer 向容器中注册自定义的Scope对象,即注册自定义的作用域实现类,关于自定义的作用域,不了解的朋友,建议先看一下:<a href="https://mp.weixin.qq.com/s?__biz=MzA5MTkxMDQ4MQ==&mid=2648933960&idx=1&sn=f4308f8955f87d75963c379c2a0241f4&chksm=88621e76bf159760d404c253fa6716d3ffce4de8df0fc1d0d5dd0cf00a81bc170a30829ee58f&token=1314297026&lang=zh_CN#rd" target="_blank">Spring系列第6篇:玩转bean scope,避免跳坑里!</a> 这个用法比较简单,定义一个`CustomScopeConfigurer`的bean就可以了,然后通过这个类来注册自定义的bean。 #### EventListenerMethodProcessor 处理`@EventListener`注解的,即spring中事件机制,需要了解spring事件的:<a href="https://mp.weixin.qq.com/s?__biz=MzA5MTkxMDQ4MQ==&mid=2648934522&idx=1&sn=7653141d01b260875797bbf1305dd196&chksm=88621044bf15995257129e33068f66fc5e39291e159e5e0de367a14e0195595c866b3aaa1972&token=1081910573&lang=zh_CN#rd" target="_blank">spring事件机制详解</a> 还有一些实现类,这里就不介绍了。 ### 使用注意 `BeanFactoryPostProcessor`接口的使用有一个需要注意的地方,在其`postProcessBeanFactory`方法中,强烈禁止去通过容器获取其他bean,此时会导致bean的提前初始化,会出现一些意想不到的问题,因为这个阶段中`BeanPostProcessor`还未准备好,本文开头4个阶段中有介绍,`BeanPostProcessor`是在第3个阶段中注册到spring容器的,而`BeanPostProcessor`可以对bean的创建过程进行干预,比如spring中的aop就是在`BeanPostProcessor`的一些子类中实现的,`@Autowired`也是在`BeanPostProcessor`的子类中处理的,此时如果去获取bean,此时bean不会被`BeanPostProcessor`处理,所以创建的bean可能是有问题的,还是通过一个案例给大家演示一下把,通透一些。 #### 来一个简单的类 ```java package com.javacode2018.lesson003.demo3.test3; import org.springframework.beans.factory.annotation.Autowired; public class UserModel { @Autowired private String name; //@1 @Override public String toString() { return "UserModel{" + "name='" + name + '\'' + '}'; } } ``` > @1:使用了@Autowired,会指定注入 ### 来个配置类 配置类中定义了2个UserModel类型的bean:user1、user2 并且定义了一个String类型的bean:name,这个会注入到UserModel中的name属性中去。 ```java package com.javacode2018.lesson003.demo3.test3; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.ComponentScan; import org.springframework.context.annotation.Configuration; @Configuration @ComponentScan public class MainConfig3 { @Bean public UserModel user1() { return new UserModel(); } @Bean public UserModel user2() { return new UserModel(); } @Bean public String name() { return "路人甲Java,带大家成功java高手!"; } } ``` #### 测试用例 > 输出容器中所有UserModel类型的bean ```java @Test public void test3() { AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(); context.register(MainConfig3.class); context.refresh(); context.getBeansOfType(UserModel.class).forEach((beanName, bean) -> { System.out.println(String.format("%s->%s", beanName, bean)); }); } ``` #### 运行输出 ```java user1->UserModel{name='路人甲Java,带大家成功java高手!'} user2->UserModel{name='路人甲Java,带大家成功java高手!'} ``` 效果不用多解释,大家一看就懂,下面来重点。 ### 添加一个BeanFactoryPostProcessor 在`postProcessBeanFactory`方法中去获取一下user1这个bean ```java package com.javacode2018.lesson003.demo3.test3; 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 { @Override public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException { beanFactory.getBean("user1"); } } ``` #### 再次运行输出 ```java user1->UserModel{name='null'} user2->UserModel{name='路人甲Java,带大家成功java高手!'} ``` 注意,user1中的name变成null了,什么情况? 是因为@Autowired注解是在`AutowiredAnnotationBeanPostProcessor`中解析的,spring容器调用`BeanFactoryPostProcessor#postProcessBeanFactory`的使用,此时spring容器中还没有`AutowiredAnnotationBeanPostProcessor`,所以此时去获取user1这个bean的时候,@Autowired并不会被处理,所以name是null。 ## 源码 ### 4个阶段的源码 4个阶段的源码为位于下面这个方法中 ```java org.springframework.context.support.AbstractApplicationContext#refresh ``` 这个方法中截取部分代码如下: ```java // 对应阶段1和阶段2:调用上下文中注册为bean的工厂处理器,即调用本文介绍的2个接口中的方法 invokeBeanFactoryPostProcessors(beanFactory); // 对应阶段3:注册拦截bean创建的bean处理器,即注册BeanPostProcessor registerBeanPostProcessors(beanFactory); // 对应阶段3:实例化所有剩余的(非延迟初始化)单例。 finishBeanFactoryInitialization(beanFactory); ``` 阶段1和阶段2的源码位于下面方法中,代码比较简单,强烈建议大家去看一下,几分钟就可以看懂了。 ```java org.springframework.context.support.PostProcessorRegistrationDelegate#invokeBeanFactoryPostProcessors(org.springframework.beans.factory.config.ConfigurableListableBeanFactory, java.util.List<org.springframework.beans.factory.config.BeanFactoryPostProcessor>) ``` ## 总结 1. 注意spring的4个阶段:bean定义阶段、BeanFactory后置处理阶段、BeanPostProcessor注册阶段、单例bean创建组装阶段 2. BeanDefinitionRegistryPostProcessor会在第一个阶段被调用,用来实现bean的注册操作,这个阶段会完成所有bean的注册 3. BeanFactoryPostProcessor会在第2个阶段被调用,到这个阶段时候,bean此时已经完成了所有bean的注册操作,这个阶段中你可以对BeanFactory中的一些信息进行修改,比如修改阶段1中一些bean的定义信息,修改BeanFactory的一些配置等等 4. 阶段2的时候,2个禁止操作:禁止注册bean、禁止从容器中获取bean 5. 本文介绍的2个接口的实现类可以通过`PriorityOrdered`接口或者`Ordered`接口来指定顺序 ## 案例源码 ```java https://gitee.com/javacode2018/spring-series ``` **路人甲java所有案例代码以后都会放到这个上面,大家watch一下,可以持续关注动态。** ## 课后作业 2个问题,欢迎留言讨论,会有红包惊喜哦! ### 问题1 大家去读一下org.springframework.context.support.AbstractApplicationContext#invokeBeanFactoryPostProcessors的源码,将内部主要的执行过程通过文字描述出来,然后留言 ### 问题2 BeanPostProcessor和BeanFactoryPostProcessors有何区别?留言 <a style="display:none" target="_blank" href="https://mp.weixin.qq.com/s/_S1DD2JADnXvpexxaBwLLg" style="color:red; font-size:20px; font-weight:bold">继续收门徒,亲手带,月薪 4W 以下的可以来找我</a> ## 最新资料 1. <a href="https://mp.weixin.qq.com/s?__biz=MzkzOTI3Nzc0Mg==&mid=2247484964&idx=2&sn=c81bce2f26015ee0f9632ddc6c67df03&scene=21#wechat_redirect" target="_blank">尚硅谷 Java 学科全套教程(总 207.77GB)</a> 2. <a href="https://mp.weixin.qq.com/s?__biz=MzkwOTAyMTY2NA==&mid=2247484192&idx=1&sn=505f2faaa4cc911f553850667749bcbb&scene=21#wechat_redirect" target="_blank">2021 最新版 Java 微服务学习线路图 + 视频</a> 3. <a href="https://mp.weixin.qq.com/s?__biz=MzkwOTAyMTY2NA==&mid=2247484573&idx=1&sn=7f3d83892186c16c57bc0b99f03f1ffd&scene=21#wechat_redirect" target="_blank">阿里技术大佬整理的《Spring 学习笔记.pdf》</a> 4. <a href="https://mp.weixin.qq.com/s?__biz=MzkwOTAyMTY2NA==&mid=2247484544&idx=2&sn=c1dfe907cfaa5b9ae8e66fc247ccbe84&scene=21#wechat_redirect" target="_blank">阿里大佬的《MySQL 学习笔记高清.pdf》</a> 5. <a href="https://mp.weixin.qq.com/s?__biz=MzkwOTAyMTY2NA==&mid=2247485167&idx=1&sn=48d75c8e93e748235a3547f34921dfb7&scene=21#wechat_redirect" target="_blank">2021 版 java 高并发常见面试题汇总.pdf</a> 6. <a href="https://mp.weixin.qq.com/s?__biz=MzkwOTAyMTY2NA==&mid=2247485664&idx=1&sn=435f9f515a8f881642820d7790ad20ce&scene=21#wechat_redirect" target="_blank">Idea 快捷键大全.pdf</a> ![](https://itsoku.oss-cn-hangzhou.aliyuncs.com/itsoku/blog/article/1/2883e86e-3eff-404a-8943-0066e5e2b454.png)
相关专辑:
Spring教程