Spring Boot源码分析 - Configuration注解
@Configuration注解
@Configuration
注解指示一个类声明一个或多个@Bean方法, 并且可以由Spring容器处理, 以在运行时为这些bean生成bean定义和服务请求.
使用ConfigurationClassParser
来对@Configuration
标注的类进行解析, 封装成ConfigurationClass
实例. 具体的实现通过ConfigurationClassPostProcessor
来实现的.
ConfigurationClassPostProcessor
实现了BeanDefinitionRegistryPostProcessor
接口, 间接实现了BeanFactorPostProcessor
接口.
#postProcessBeanDefinitionRegistry()
: 注册所有ConfigurationClass
中的BeanDefinition
, 包括@Bean
注解的方法,@ImporResource
引入的资源中定义的bean, 和@Import
注解引入的ImportBeanDefinitionRegistrar
中注册的BeanDefinition
#postProcessBeanFactory()
: 在运行时以通过cglig
增强的类来替换ConfigurationClass
, 为服务bean请求做准备. 增强的实现是通过ConfigurationClassEnhancer
完成的.
插入一点, ConfigurationClassEnhancer
实现了直接使用bean注册方法来获取bean的操作, 提供了一个BeanMethodInterceptor
的内部类来实行.
@Configuration
public class Config {
@Bean
public A a() {
...
return a;
}
@Bean
public B b() {
b.setA(a());
...
return b;
}
}
Full ConfigurationClass VS Lite ConfigurationClass
先说区别: full的ConfigurationClass
会使用CGLIB进行增强.
查看类ConfigurationClassUtils
, 其中有两个方法#isFullConfigurationClass()
和#isLiteConfigurationClass()
.
方法的实现是去检查BeanDefinition
中的ConfigurationClassPostProcessor.configurationClass
属性, 是full
还是lite
.
这个属性的值又来源于#checkConfigurationClassCandidate()
方法, 如果BeanDefinition
使用的是@Configuration
注解, 则为full
; 如果是@Component
, @ComponentScan
, @Import
或者@ImportResource
中的任何一种, 则为lite
. 如果是ConfigurationClass
, 则会继续为其添加顺序属性.
public static boolean checkConfigurationClassCandidate(BeanDefinition beanDef, MetadataReaderFactory metadataReaderFactory) {
...
if (isFullConfigurationCandidate(metadata)) {
beanDef.setAttribute(CONFIGURATION_CLASS_ATTRIBUTE, CONFIGURATION_CLASS_FULL);
}
else if (isLiteConfigurationCandidate(metadata)) {
beanDef.setAttribute(CONFIGURATION_CLASS_ATTRIBUTE, CONFIGURATION_CLASS_LITE);
}
else {
return false;
}
// It's a full or lite configuration candidate... Let's determine the order value, if any.
Map<String, Object> orderAttributes = metadata.getAnnotationAttributes(Order.class.getName());
if (orderAttributes != null) {
beanDef.setAttribute(ORDER_ATTRIBUTE, orderAttributes.get(AnnotationUtils.VALUE));
}
}