title: Spring 源码解析 —— 配置类处理流程(@Configuration)
date: 2021/01/14 13:36
remark: Spring 版本为 5.2.5
简介
@Configuration 注解的主要作用就是向容器中注入一些 bean,所以很容易想到 Spring 是通过 BeanFactoryPostProcessor 来实现对配置类进行处理的,处理类为 ConfigurationClassPostProcessor,我们看一下他的体系图:
data:image/s3,"s3://crabby-images/5ac10/5ac1029d0e5597bdf34c92fa5775429f16b5ee99" alt=""
ConfigurationClassPostProcessor 实现了 BeanDefinitionRegistryPostProcessor 在刷新容器的时候调用它的 postProcessBeanDefinitionRegistry() 方法。
data:image/s3,"s3://crabby-images/fb8ad/fb8add54fb7e4598ad53c09b56390f6bdc5e7561" alt=""
我们先不着急继续往下看代码,先看下下面的图简单了解一下这个方法的整个的流程:
data:image/s3,"s3://crabby-images/e4de4/e4de4f817e9099909634de527b6c908d6cb44936" alt=""
开始吧 ConfigurationClassPostProcessor#processConfigBeanDefinitions
本部分测试demo,建议先看一下 demo,这样更好理解。
data:image/s3,"s3://crabby-images/f6137/f61376094f2d20cfa52b32125769730f247c69da" alt=""
data:image/s3,"s3://crabby-images/dbbee/dbbeed80380585014e7042e9d159602a594531fe" alt=""
tag1 parser.parse(candidates)
data:image/s3,"s3://crabby-images/1f831/1f83133b463d926edb11a86f10c18267c50c1162" alt=""
data:image/s3,"s3://crabby-images/6afcd/6afcdf7cf36a53270037e866ba4398f0e055863e" alt=""
tag1.1 conditionEvaluator.shouldSkip
data:image/s3,"s3://crabby-images/5c0e8/5c0e8bb0e52ffe59c9fafb00bc821f1689dcb590" alt=""
tag1.2 doProcessConfigurationClass(configClass, sourceClass, filter)
data:image/s3,"s3://crabby-images/6d6cb/6d6cba092399727123d79c0b0924b071e3adc4d3" alt=""
data:image/s3,"s3://crabby-images/bd60b/bd60be2a0d7cc97bee555d65095ca21f5cc2691c" alt=""
@ComponentScan 会直接将扫描到的实体加入容器中。
data:image/s3,"s3://crabby-images/8554b/8554b44880b24f323ee134e3af152feb1943b1c4" alt=""
data:image/s3,"s3://crabby-images/052ca/052ca2e92978ba2c8e45703950aa2d4f64f58895" alt=""
tag1.2.1 processMemberClasses(configClass, sourceClass, filter)
data:image/s3,"s3://crabby-images/f8a20/f8a2058c9c8cb15d818cd31f1c38f4973d211640" alt=""
tag1.2.2 processPropertySource(propertySource)
data:image/s3,"s3://crabby-images/9e56a/9e56a5f27e870896e9ea889282afc218b66098de" alt=""
data:image/s3,"s3://crabby-images/78eb0/78eb0f0441833e005986c33eaeaffc60d11f43c8" alt=""
tag1.2.3 processImports(configClass, sourceClass, getImports(sourceClass), filter, true);
先看下 getImports(sourceClass) 方法:
private Set<SourceClass> getImports(SourceClass sourceClass) throws IOException {
Set<SourceClass> imports = new LinkedHashSet<>();
Set<SourceClass> visited = new LinkedHashSet<>();
collectImports(sourceClass, imports, visited);
return imports;
}
// 递归的获取当前配置类上的 @Import 注解(因为注解上也可能有 @Import 注解)
private void collectImports(SourceClass sourceClass, Set<SourceClass> imports, Set<SourceClass> visited)
throws IOException {
if (visited.add(sourceClass)) {
for (SourceClass annotation : sourceClass.getAnnotations()) {
String annName = annotation.getMetadata().getClassName();
if (!annName.equals(Import.class.getName())) {
collectImports(annotation, imports, visited);
}
}
imports.addAll(sourceClass.getAnnotationAttributes(Import.class.getName(), "value"));
}
}
data:image/s3,"s3://crabby-images/81537/81537bd536da93ea8640355e0049d1473ede1287" alt=""
data:image/s3,"s3://crabby-images/06178/061788e1c337888598ffd430e699b0a6a00ef9ba" alt=""
注意:此时引入的新的配置类的 bd (除 @ComponentScan 扫描进来的)还没有放进 bdRegistry 中,只是存放在了 ConfigurationClassParser.configurationClasses 中了,之后要由 reader 将其 bd 放进 bdRegistry 中
tag2 parser.validate()
data:image/s3,"s3://crabby-images/cef51/cef51caa5277e91db6596d9f7585f9bc603286ec" alt=""
tag3 reader.loadBeanDefinitions(configClasses)
data:image/s3,"s3://crabby-images/ac1f6/ac1f6d25a9a8bffd697e88ee9db3726d5595afeb" alt=""
tag3.1 registerBeanDefinitionForImportedConfigurationClass(configClass);
data:image/s3,"s3://crabby-images/8748d/8748d5395bab95ec935293e0371cf3f9d6041412" alt=""
tag3.2 loadBeanDefinitionsForBeanMethod(beanMethod);
data:image/s3,"s3://crabby-images/74fd5/74fd5294864e5aad22940d387b9db519d28f39f1" alt=""
data:image/s3,"s3://crabby-images/b0296/b0296b5ca5b4848a0e7625d14da19c0a33e947db" alt=""
data:image/s3,"s3://crabby-images/9fadc/9fadc1e0e9a845bad899e1ef61f0f45dcc482f75" alt=""
tag3.3 loadBeanDefinitionsFromRegistrars(configClass.getImportBeanDefinitionRegistrars());
简单到无以复加
private void loadBeanDefinitionsFromRegistrars(Map<ImportBeanDefinitionRegistrar, AnnotationMetadata> registrars) {
registrars.forEach((registrar, metadata) ->
registrar.registerBeanDefinitions(metadata, this.registry, this.importBeanNameGenerator));
}
冷知识:Full模式和Lite模式
你真的懂Spring Java Config 吗?Full @Configuration vs lite @Bean mode
data:image/s3,"s3://crabby-images/9e848/9e848fb20c00097f691ed93036de898ff5cc31f4" alt=""
他把标注这 4 个注解作为配置类我可以理解(有可能是通过配置文件或其他方式注入的 bean),但搞不懂为啥 spring 要给普通的标注那 4 个注解(没有@Bean)方法的 Lite 模式,Lite 模式作用于的是 @Bean 标注的方法之间互相引用,对这 4 个注解好像没啥用吧。可能是处理过的总要给一个模式,来告诉 spring 已经处理过这个类了,所以就把 Lite 作为默认模式了。
网友评论