通过这篇文章来大家分享一下,另外一个Springboot的扩展点BeanDefinitionRegistryPostProcessor,一般称这类扩展点为容器级后置处理器,另外一类是Bean级的后置处理器;容器级的后置处理器会在Spring容器初始化后、刷新前这个时间执行一次,Bean级的后置处理器,则是在每一个Bean实例化前后都会执行。
图片
public interface BeanDefinitionRegistryPostProcessor extends BeanFactoryPostProcessor { void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException;}
@FunctionalInterfacepublic interface BeanFactoryPostProcessor { void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException;}
总结起来就是,在所有的BeanDefinition加载完成之后,Bean真正被实例化之前,可以通过实现BeanDefinitionRegistryPostProcessor接口,对BeanDefinition再做一些定制化的操作,比如修改某个bean的BeanDefinition的属性、手动注册一些复杂的Bean。
对于Spring原理不太熟悉的小伙伴心里看到这可能有点晕了,BeanDefinition是什么?BeanDefinitionRegistry又是什么?ConfigurableListableBeanFactory又又是什么?别着急,这里拐个弯简单的解释一下,方便下面的内容理解起来更顺畅。
大家都知道,Spring的核心之一是IOC(控制反转),Spring之所以可以实现bean控制权的反转,是因为Spring的容器功能,在bean纳入Spring容器管理前,所有bean会被抽象封装成一个BeanDefinition实例,然后会在不同的时机根据BeanDefinition实例信息对bean进行实例化。
简单说,Dog.java描述狗这一类动物的属性和行为,BeanDefinition描述Dog.java这个类。
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);}
上面提到了Spring的容器,Spring的核心之一是IOC,那么Spring的容器设计就是核心中的核心了。Spring的容器有多种形态,最基础的形态就是BeanFactory,ConfigurableListableBeanFactory间接继承了BeanFactory,因此ConfigurableListableBeanFactory实现类除了有Spring基础版本容器的功能外,还有一些高级的功能,Springboot默认的实际实现是DefaultListableBeanFactory,有兴趣的小伙伴可以以此为入口深入探究一番,这里不展开细说了。
图片
下面通过一个具体类MyBeanDefinitionRegistryPostProcessor实现BeanDefinitionRegistryPostProcessor接口,来探究BeanDefinitionRegistryPostProcessor实现类的初始化和执行过程。
@Datapublic class Dog { private String name; private String color;}
@Componentpublic 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------------"); }}
单元测试
@Testpublic void test(){ AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext("com.fanfu"); Dog dog = ((Dog) context.getBean("dog")); System.out.println(dog.getName()); System.out.println(dog.getColor());}
通过BeanDefinitionRegistryPostProcessorUML类图可以看出BeanDefinitionRegistryPostProcessor继承了BeanFactoryPostProcessor,postProcessBeanDefinitionRegistry()方法属于BeanDefinitionRegistryPostProcessor,postProcessBeanFactory()属于BeanFactoryPostProcessor,所有实现了BeanDefinitionRegistryPostProcessor接口的实现类都需要实现这个方法,而作为Springboot的扩展点之一,其扩展的逻辑也在这两个方法中;
图片
通过自定义的MyBeanDefinitionRegistryPostProcessor类,实现BeanDefinitionRegistryPostProcessor接口,从项目启动开始,其执行过程如下:
下面是我根据整个调用过程画的一个时序图,过程确实比较复杂,但是逻辑比较清晰,因此并不难理解,想要真的搞清楚整个过程,最好的方法就是照着这个图,亲自执行一遍,通过debug观察每一个关键节点的执行过程。
图片
spring-boot-starter-web中内置的实现类有CachingMetadataReaderFactoryPostProcessor、ConfigurationClassPostProcessor、ConfigurationWarningsPostProcessor、EmbeddedDataSourceBeanFactoryPostProcessor、ImportsCleanupPostProcessor、TestRestTemplateRegistrar、WebTestClientRegistrar、WsdlDefinitionBeanFactoryPostProcessor,观察一下每个实现类会发现:都比较类似,这些内置实现类都是Springboot中的内部类,通过这些BeanDefinitionRegistryPostProcessor内部实现类向Spring容器中注册了一些特殊的BeanDefinition,如果展开详细再说一说这些Bean,怕是一天一夜也说不完,有兴趣的小伙伴可以深入了解一下,这里就不再展开了。
通过梳理整个过程,其实最关键的就是一句话:在Spring容器初始后、未刷新前,即Bean已被扫描注册为BeanDefinition后,未正式实例化前,可以通过实现BeanDefinitionRegistryPostProcessor做一些额外的操作。
本文链接://www.dmpip.com//www.dmpip.com/showinfo-26-34657-0.htmlSpringboot扩展点之BeanDefinitionRegistryPostProcessor,你学会了吗?
声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com
上一篇: 一个注解搞定多数据源切换,你学会了吗?
下一篇: 100行代码实现审计日志中间件