Springboot扩展点之BeanDefinitionRegistryPostProcessor

文章详细介绍了Springboot的扩展点BeanDefinitionRegistryPostProcessor的工作原理和实现方式,它在Spring容器初始化后、刷新前执行,允许对BeanDefinition进行增删改查。内容包括BeanDefinition的定义、BeanDefinitionRegistry的作用以及ConfigurableListableBeanFactory的解释。此外,文章还提供了一个自定义实现示例,并分析了初始化和执行时机,以及内置实现类的用途。

Springboot扩展点系列实现方式、工作原理集合:

Springboot扩展点之ApplicationContextInitializer

Springboot扩展点之BeanFactoryPostProcessor

Springboot扩展点之BeanDefinitionRegistryPostProcessor

Springboot扩展点之BeanPostProcessor

Springboot扩展点之InstantiationAwareBeanPostProcessor

Springboot扩展点之SmartInstantiationAwareBeanPostProcessor

Springboot扩展点之ApplicationContextAwareProcessor

Springboot扩展点之@PostConstruct

Springboot扩展点之InitializingBean

Springboot扩展点之SmartInitializingSingleton

Springboot扩展点之CommandLineRunner和ApplicationRunner

Springboot扩展点之FactoryBean

Springboot扩展点之DisposableBean

Springboot扩展点系列之终结篇:Bean的生命周期


前言

通过这篇文章来大家分享一下,另外一个Springboot的扩展点BeanDefinitionRegistryPostProcessor,一般称这类扩展点为容器级后置处理器,另外一类是Bean级的后置处理器;容器级的后置处理器会在Spring容器初始化后、刷新前这个时间执行一次,Bean级的重置处理器,则是在每一个Bean实例化前后都会执行。

1. 功能特性

  1. postProcessBeanDefinitionRegistry()方法可以通过BeanDefinitionRegistry对BeanDefintion进行增删改查;

  1. 继承了BeanFactoryPostProcessor,BeanFactoryPostProcessor是容器级别的扩展接口,org.springframework.beans.factory.config.BeanFactoryPostProcessor#postProcessBeanFactory方法在容器实例化后、刷新容器前被执行,即在容器刷新前还可以对BeanDefintion再作一些操作;

public interface BeanDefinitionRegistryPostProcessor extends BeanFactoryPostProcessor {
   void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException;
}
@FunctionalInterface
public interface BeanFactoryPostProcessor {
   void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException;
}

总结起来就是,在所有的BeanDefinition加载完成之后,Bean真正被实例化之前,可以通过实现BeanDefinitionRegistryPostProcessor接口,对BeanDefinition再做一些定制化的操作,比如修改某个bean的BeanDefinition的属性、手动注册一些复杂的Bean。

对于Spring原理不太熟悉的小伙伴心里看到这可能有点晕了,BeanDefinition是什么?BeanDefinitionRegistry又是什么?ConfigurableListableBeanFactory又又是什么?别着急,这里拐个弯简单的解释一下,对整篇文章的理解会更顺畅。

1.1 BeanDefinition

大家都知道,Spring的核心之一是IOC(控制反转),Spring之所以可以实现bean控制权的反转,是因为Spring的容器功能,在bean纳入Spring容器管理前,所有bean会被抽象封装成一个BeanDefinition实例,然后会在不同的时机根据BeanDefinition实例信息对bean进行实例化。

简单说,Dog.java描述狗这一类动物的属性和行为,BeanDefinition描述Dog.java这个类。

1.2 BeanDefinitionRegistry

BeanDefinitionRegistry从字面意思看是bean的定义信息的注册登记,其实这个类的功能和字面意思一样,就是对BeanDefinition进行管理(增删改查);

public interface BeanDefinitionRegistry extends AliasRegistry {
    //注册beanDefinition
   void registerBeanDefinition(String beanName, BeanDefinition beanDefinition)
         throws BeanDefinitionStoreException;
    //移除指定的beanDefinition
   void removeBeanDefinition(String beanName) throws NoSuchBeanDefinitionException;
    //根据beanName查询beanDefinition
   BeanDefinition getBeanDefinition(String beanName) throws NoSuchBeanDefinitionException;
    //判断某个beanDefinition是否已经注册
   boolean containsBeanDefinition(String beanName);
    //获取所有已注册的beanDefinition
   String[] getBeanDefinitionNames();
    //获取所有已注册的beanDefinition的数量
   int getBeanDefinitionCount();
    //判断某个beanDefinition是否已经被使用
   boolean isBeanNameInUse(String beanName);
}

1.3 ConfigurableListableBeanFactory

上面提到了Spring的容器,Spring的核心之一是IOC,那么Spring的容器设计就是核心中的核心了。Spring的容器有多种形态,最基础的形态就是BeanFactory,ConfigurableListableBeanFactory间接继承了BeanFactory,因此ConfigurableListableBeanFactory实现类除了有Spring基础版本容器的功能外,还有一些高级的功能,Springboot默认的实际实现是DefaultListableBeanFactory,有兴趣的小伙伴可以以此为入口深入探究一番,这里不展开细说了。

2.自定义实现

2.1 MyBeanDefinitionRegistryPostProcessor

下面通过一个具体类MyBeanDefinitionRegistryPostProcessor实现BeanDefinitionRegistryPostProcessor接口,来探究BeanDefinitionRegistryPostProcessor实现类的初始化和执行过程。

  1. 在postProcessBeanDefinitionRegistry()方法被调用的时候手工在Spring中注册了Dog类的BeanDefinition信息;

  1. 在postProcessBeanFactory()方法被调用的时候,从Spring容器中取出Dog类的BeanDefinition信息和Dog类的实例;

@Data
public class Dog {
    private String name;
    private String color;
}
@Component
public class MyBeanDefinitionRegistryPostProcessor implements BeanDefinitionRegistryPostProcessor {
    @Override
    public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException {
        //手工定义一个beanDefinition实例
        RootBeanDefinition beanDefinition = new RootBeanDefinition();
        //给beanDefinition填充属性
        beanDefinition.setBeanClass(Dog.class);
        MutablePropertyValues propertyValues = new MutablePropertyValues();
        PropertyValue propertyValue1 = new PropertyValue("name", "旺财");
        PropertyValue propertyValue2 = new PropertyValue("color", "黑色");
        propertyValues.addPropertyValue(propertyValue1);
        propertyValues.addPropertyValue(propertyValue2);
        beanDefinition.setPropertyValues(propertyValues);
        //注册手工定义的beanDefinition
        registry.registerBeanDefinition("dog", beanDefinition);
    }
    @Override
    public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
        System.out.println("-----------start------------");
        //根据类名取出手工注册的beanDefinition
        BeanDefinition beanDefinition = beanFactory.getBeanDefinition("dog");
        System.out.println(beanDefinition.getBeanClassName());
        //根据类从容器中取出手工注册的beanDefinition所描述的实例bean
        Dog dog = beanFactory.getBean(Dog.class);
        System.out.println(dog.getName());
        System.out.println(dog.getColor());
        System.out.println("-----------end------------");
    }
}

单元测试

@Test
public void test(){
    AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext("com.fanfu");
    Dog dog = ((Dog) context.getBean("dog"));
    System.out.println(dog.getName());
    System.out.println(dog.getColor());
}

2.2 UML类图

通过BeanDefinitionRegistryPostProcessorUML类图可以看出BeanDefinitionRegistryPostProcessor继承了BeanFactoryPostProcessor,postProcessBeanDefinitionRegistry()方法属于BeanDefinitionRegistryPostProcessor,postProcessBeanFactory()属于BeanFactoryPostProcessor,所有实现了BeanDefinitionRegistryPostProcessor接口的实现类都需要实现这两个方法,而作为Springboot的扩展点之一,其扩展的逻辑也在这两个方法中。

3. 初始化和执行时机

通过自定义的MyBeanDefinitionRegistryPostProcessor类,实现BeanDefinitionRegistryPostProcessor接口,从项目启动开始,其执行过程如下:

  1. 执行项目的主类,org.springframework.boot.SpringApplication#run被调用;

  1. 进入boot.SpringApplication#run方法后,刚开始是一些Spring容器初始化的配置操作,直到执行到org.springframework.boot.SpringApplication#refreshContext,开始容器刷新,进入了关键阶段;

  1. 在SpringApplication#refreshContext,实际的刷新逻辑是在org.springframework.context.support.AbstractApplicationContext#refresh方法中;

  1. AbstractApplicationContext#refresh方法中,调用org.springframework.context.support.AbstractApplicationContext#invokeBeanFactoryPostProcessors开始初始化和执行实现BeanDefinitionRegistryPostProcessor接口的postProcessBeanDefinitionRegistry()和postProcessBeanFactory();

  1. 进入AbstractApplicationContext#invokeBeanFactoryPostProcessors方法,发现又调用了org.springframework.context.support.PostProcessorRegistrationDelegate#invokeBeanFactoryPostProcessors();

  1. 在PostProcessorRegistrationDelegate#invokeBeanFactoryPostProcessors()方法中,并不是直接就初始化和执行postProcessBeanDefinitionRegistry()和postProcessBeanFactory(),而是又进行了一系列的判断,其判断顺序是:1、通过AbstractApplicationContext#addBeanFactoryPostProcessor提前注册的BeanDefinitionRegistryPostProcessor实现类;2、实现了PriorityOrdered接口;3、是否实现了Ordered;4、剩下的其他BeanDefinitionRegistryPostProcessor实现类;自定义的MyBeanDefinitionRegistryPostProcessor就属于第4类,所以是所有实现里较晚才被执行的,如果想要提前被执行,可以考虑前面三种方式;

  1. 在PostProcessorRegistrationDelegate#invokeBeanFactoryPostProcessors()方法中执行完MyBeanDefinitionRegistryPostProcessor#postProcessBeanDefinitionRegistry方法后,紧接着就开始执行MyBeanDefinitionRegistryPostProcessor#postProcessBeanFactory方法了;从整个调用过程看postProcessBeanDefinitionRegistry()是早于postProcessBeanFactory()方法执行;

下面是我根据整个调用过程画的一个时序图,过程确实比较复杂,但是逻辑比较清晰,因此并不难理解,想要真的搞清楚整个过程,最好的方法就是照着这个图,亲自执行一遍,通过debug观察每一个关键节点的执行过程。

4. 内部实现类

spring-boot-starter-web中内置的实现类有CachingMetadataReaderFactoryPostProcessor、ConfigurationClassPostProcessor、ConfigurationWarningsPostProcessor、EmbeddedDataSourceBeanFactoryPostProcessor、ImportsCleanupPostProcessor、TestRestTemplateRegistrar、WebTestClientRegistrar、WsdlDefinitionBeanFactoryPostProcessor,观察一下每个实现类会发现:都比较类似,这些内置实现类都是Springboot中的内部类,通过这些BeanDefinitionRegistryPostProcessor内部实现类向Spring容器中注册了一些特殊的BeanDefinition,如果展开详细再说一说这些Bean,怕是一天一夜也说不完,有兴趣的小伙伴可以深入了解一下,这里就不再展开了。

5. 总结

通过梳理整个过程,其实最关键的就是一句话:在Spring容器初始后、未刷新前,即Bean已被扫描注册为BeanDefinition后,未正式实例化前,可以通过实现BeanDefinitionRegistryPostProcessor做一些额外的操作。

<think>我们正在讨论SpringBoot中的BeanFactoryPostProcessor。根据用户的问题和提供的引用,我们需要解释BeanFactoryPostProcessor的作用及使用场景。注意,用户之前询问的是BeanPostProcessor,现在转向BeanFactoryPostProcessor,因此要明确区分两者。根据引用[1][2][3][5]:-BeanFactoryPostProcessor是Spring框架的一个扩展接口,它允许在Spring容器加载了Bean定义(BeanDefinition)之后,但在Bean实例化之前,对Bean的定义进行修改。-与BeanPostProcessor(作用于Bean实例化后的生命周期)不同,BeanFactoryPostProcessor作用于容器级,修改的是Bean的元数据(BeanDefinition)。作用:1.修改Bean定义:可以修改Bean的属性值、构造函数参数值,甚至替换Bean定义(例如改变Bean的类或作用域)[^1][^5]。2.注册新的Bean定义:通过BeanDefinitionRegistryPostProcessor(BeanFactoryPostProcessor的子接口)可以动态注册新的Bean定义[^2]。3.处理配置占位符:如PropertyPlaceholderConfigurer(已过时,现在推荐使用PropertySourcesPlaceholderConfigurer)用于解析属性文件中的占位符,并将其注入到Bean定义中[^1]。4.条件化注册Bean:在SpringBoot中,很多自动配置类使用BeanFactoryPostProcessor来根据条件动态调整Bean定义。使用场景:-属性占位符解析:将外部配置(如application.properties)中的值注入到Bean定义的占位符中[^1]。-配置类处理:SpringBoot的@Configuration类处理过程中,会使用BeanFactoryPostProcessor(如ConfigurationClassPostProcessor)来解析配置类,注册其中定义的Bean[^3]。-中间件集成:在集成第三方库时,可能需要动态修改Bean定义。例如,在MyBatis-Spring集成中,MapperScannerConfigurer(实现了BeanDefinitionRegistryPostProcessor)扫描指定包并注册Mapper接口的Bean定义[^2]。-性能调优:引用[4]虽然讲的是性能调优,但BeanFactoryPostProcessor的使用也可能涉及性能优化,例如通过减少不必要的Bean定义来优化容器启动速度。实现机制:-Spring容器启动时,在创建BeanFactory之后,会调用所有BeanFactoryPostProcessor的postProcessBeanFactory方法(对于BeanDefinitionRegistryPostProcessor,会先调用其postProcessBeanDefinitionRegistry方法)[^2][^3]。-这些处理器按优先级执行(通过Ordered接口或@Order注解控制)。示例(引用[5]):在applyPropertyValue方法中,通过BeanFactory获取BeanDefinition,然后修改其属性值。这展示了如何修改Bean定义。总结:BeanFactoryPostProcessor是SpringBoot中用于在容器初始化阶段修改Bean定义的核心扩展点,常用于属性解析、动态注册Bean等场景[^1][^2][^3][^5]。现在,按照要求生成相关问题。</think>### Spring Boot中BeanFactoryPostProcessor的作用及使用场景 `BeanFactoryPostProcessor`是Spring Boot框架的核心扩展接口,作用于**容器初始化阶段**,主要用于修改或增强Bean的元数据(BeanDefinition)。它与`BeanPostProcessor`(作用于Bean实例化阶段)形成互补,共同构成Spring容器的扩展机制[^1][^3][^5]。 --- #### 一、核心作用 1. **修改Bean定义(BeanDefinition)** 在Bean实例化**之前**,可动态修改Bean的属性值、构造函数参数、作用域等元数据[^1][^5]。 ```java public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) { BeanDefinition bd = beanFactory.getBeanDefinition("myBean"); bd.getPropertyValues().add("timeout", 5000); // 修改属性值 } ``` 2. **动态注册新Bean** 通过子接口`BeanDefinitionRegistryPostProcessor`(继承自`BeanFactoryPostProcessor`)可注册新Bean定义[^2][^3]。 ```java public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) { registry.registerBeanDefinition("newBean", new RootBeanDefinition(NewService.class)); } ``` 3. **处理配置占位符** 如`PropertySourcesPlaceholderConfigurer`解析`${}`占位符,将外部配置(如`application.properties`)注入Bean定义[^1][^5]。 4. **条件化Bean注册** 结合`@Conditional`注解实现条件化配置,例如根据环境变量动态启用/禁用Bean[^3]。 --- #### 二、典型使用场景 1. **外部配置集成** - 解析属性文件(如`application.yml`)并应用到Bean定义中 - 示例:`PropertyPlaceholderConfigurer`(已过时)或`PropertySourcesPlaceholderConfigurer`[^1] 2. **自动配置类处理** Spring Boot自动配置的核心机制: - `ConfigurationClassPostProcessor`扫描`@Configuration`类 - 解析`@Bean`、`@Import`等注解,注册Bean定义[^3][^5] 3. **中间件集成** - 第三方库(如MyBatis、Hibernate)通过`BeanDefinitionRegistryPostProcessor`动态注册组件 - 示例:MyBatis的`MapperScannerConfigurer`扫描Mapper接口[^2] 4. **性能优化** - 预加载时修改Bean定义,减少运行时开销 - 示例:缓存Bean元数据、延迟初始化配置[^4] 5. **安全增强** - 加密敏感属性(如数据库密码),在Bean定义阶段解密 ```java bd.getPropertyValues().add("password", decrypt(encryptedValue)); ``` --- #### 三、执行时机与机制 1. **生命周期阶段** ```mermaid graph LR A[加载Bean定义] --> B[执行BeanFactoryPostProcessor] B --> C[实例化Bean] C --> D[执行BeanPostProcessor] ``` 2. **执行顺序** - 优先执行`BeanDefinitionRegistryPostProcessor`(子接口) - 再执行普通`BeanFactoryPostProcessor` - 顺序可通过`Ordered`接口或`@Order`注解控制[^2][^3] 3. **源码关键点** - 入口:`AbstractApplicationContext.refresh()`中的`invokeBeanFactoryPostProcessors()` - 核心实现:`PostProcessorRegistrationDelegate.invokeBeanFactoryPostProcessors()`[^5] --- #### 四、与BeanPostProcessor的区别 | **特性** | BeanFactoryPostProcessor | BeanPostProcessor | |------------------------|---------------------------------------------|---------------------------------------| | **作用阶段** | Bean实例化前(修改Bean定义) | Bean实例化后(修改Bean实例) | | **操作对象** | BeanDefinition(元数据) | Bean实例对象 | | **典型应用** | 占位符解析、动态注册Bean | 依赖注入、AOP代理、生命周期回调 | | **子接口扩展** | BeanDefinitionRegistryPostProcessor | InstantiationAwareBeanPostProcessor | > 引用说明: > [^1]: 修改Bean定义的核心机制,如属性占位符解析 > [^2]: 子接口`BeanDefinitionRegistryPostProcessor`的动态注册能力 > [^3]: 在Spring Boot自动配置中的实际应用 > [^5]: 源码级修改Bean定义的示例(如`applyPropertyValue`方法)
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

凡夫贩夫

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值