Spring提供了一种叫做BeanFactoryPostProcessor的容器扩展机制。该机制允许我们在容器实例化相应对象之前,对注册到容器的BeanDefinition所保存的信息做相应的修改。这就相当于在容器实现的第一阶段最后加入一道工序,让我们对最终的BeanDefinition做一些额外的操作,比如修改其中bean定义的某些属性,为bean定义增加其他信息等。 (转自 《Spring揭秘》)
我们隆重介绍的ConfigurationClassPostProcessor就是一种BeanFactoryPostProcessor。该类的完整签名如下:
public class ConfigurationClassPostProcessor implements BeanDefinitionRegistryPostProcessor, PriorityOrdered, ResourceLoaderAware, BeanClassLoaderAware, EnvironmentAware
ConfigurationClassPostProcessor 初始化
BeanDefinitionRegistryPostProcessor 间接实现了BeanFactoryPostProcessor 并对其进行了一定的扩展,使得实现该接口的类能够注册bean definitions。
SpringBoot 在创建AnnotationConfigServletWebServerApplicationContext的过程中,一并初始化了AnnotatedBeanDefinitionReader。同时AnnotatedBeanDefinitionReader 内部通过AnnotationConfigUtils初始化了ConfigurationClassPostProcessor。此刻AnnotationConfigServletWebServerApplicationContext 具有了扫描注解的功能。AnnotationConfigServletWebServerApplicationContext 实际上利用RootBeanDefinition的克隆进行扫描。
RootBeanDefinition def = new RootBeanDefinition(ConfigurationClassPostProcessor.class);
ConfigurationClassPostProcessor 扫描实例
AnnotationConfigServletWebServerApplicationContext 在refresh的时候,通过调用factory的beanFactoryPostProcessors,实现对beanDefinition的改变,而ConfigurationClassPostProcessor 便赫然在列。
ConfigurationClassPostProcessor 在处理的过程中,获取到启动类的元信息。
Set<AnnotationAttributes> componentScans = AnnotationConfigUtils.attributesForRepeatable( sourceClass.getMetadata(), ComponentScans.class, ComponentScan.class);
经过
Set<BeanDefinitionHolder> scannedBeanDefinitions = this.componentScanParser.parse(componentScan,sourceClass.getMetadata().getClassName());
获取到需要生成beanDefinition的数据元信息,并注册及进去。
Set<BeanDefinition> candidates = findCandidateComponents(basePackage);
registerBeanDefinition(definitionHolder, this.registry);