SpringBoot 源码分析 - 自动配置深度分析一
- refresh和自动配置大致流程
- 如何自动配置
- SpringBootApplication注解
- EnableAutoConfiguration注解
- AutoConfigurationImportSelector自动配置导入选择器
- DeferredImportSelectorHandler的handle
- DeferredImportSelectorGroupingHandler的register注册DeferredImportSelectorHolder
- DeferredImportSelectorGrouping
refresh和自动配置大致流程
如何自动配置
我们看下这段代码,红色框是什么放入的注册类,其实这就是我们要注册的配置类,自动装配跟他有关:
public static void main(String[] args) {SpringApplication.run(SpringBootLearnApplication.class, args);}
为什么呢,因为他头上有注解 SpringBootApplication
:
@SpringBootApplication
public class SpringBootLearnApplication
所以其实你只要传一个头上有这个注解的类就可以了,不一定要是main
方法的类。
SpringBootApplication注解
那为什么就有这个注解就能自动配置呢,我们来看看这个注解,其他你可以不管,但是EnableAutoConfiguration
你得看:
首先这个是一个配置类,可以被解析,因为有SpringBootConfiguration
注解,被Configuration
注解了:
这样我们的SpringBootApplication
就类似于我们经常配置的java
配置类。有了这个前提,我们就可以先看这个配置类的解析过程,其实就是ConfigurationClassPostProcessor
的解析处理,解析流程我以前的spring
源码文章都讲过,其中processImports
的处理也特别讲过,会先递归获取所有import
进来的类,然后按不同类型进行判断处理,重点就是这里啦,我们先来看下EnableAutoConfiguration
注解是不是有import
呢。
EnableAutoConfiguration注解
有个AutoConfigurationImportSelector
被import
,其实还有一个在AutoConfigurationPackage
注解里,暂时不说,重点是AutoConfigurationImportSelector
。
AutoConfigurationImportSelector自动配置导入选择器
看下他的结构,左边不用管,就是为了回调拿到一些属性,好对容器操作,主要是右边,我们前面spring
的文章讲过ImportSelector
的原理,想看的朋友可以看下这篇文章,有这个基础对下面的理解比较好,因为这里还有个DeferredImportSelector
,他会在ConfigurationClassParser
解析方法的最后来处理。
可以看到,他被处理的时候:
DeferredImportSelectorHandler
就是专门来处理这些DeferredImportSelector
的。
DeferredImportSelectorHandler的handle
首先会封装一个持有器DeferredImportSelectorHolder
,如果deferredImportSelectors
为空,表示在被DeferredImportSelectorGroupingHandler
处理中,DeferredImportSelectorGroupingHandler
又是什么呢,其实我们要处理的DeferredImportSelectorHolder
会根据Group
进行分组,来区分不同的ImportSelector
,比如我们这个自动装配的AutoConfigurationImportSelector
,是属于AutoConfigurationGroup
类型的,分组处理方便,到时候只需要遍历所有组中的ImportSelector
统一处理即可。因为处理的方式是迭代器循环,所以不能添加,只能直接处理,当然如果不为空,表示没有在处理,可以添加到deferredImportSelectors
集合里。
public void handle(ConfigurationClass configClass, DeferredImportSelector importSelector) {DeferredImportSelectorHolder holder = new DeferredImportSelectorHolder(configClass, importSelector);if (this.deferredImportSelectors == null) {//直接处理DeferredImportSelectorGroupingHandler handler = new DeferredImportSelectorGroupingHandler();handler.register(holder);handler.processGroupImports();}else {this.deferredImportSelectors.add(holder);//添加}}
DeferredImportSelectorGroupingHandler的register注册DeferredImportSelectorHolder
可以看到组处理器里面会有一个groupings
映射,就是来存放Group
类别和DeferredImportSelectorGrouping
映射的,同一个Group
类别里面可以很多个DeferredImportSelectorGrouping
,因为DeferredImportSelectorGrouping
里面有集合。
//自动装配类型和组映射private final Map<Object, DeferredImportSelectorGrouping> groupings = new LinkedHashMap<>();//注解属性和配置类的映射private final Map<AnnotationMetadata, ConfigurationClass> configurationClasses = new HashMap<>();//注册分组public void register(DeferredImportSelectorHolder deferredImport) {Class<? extends Group> group = deferredImport.getImportSelector().getImportGroup();DeferredImportSelectorGrouping grouping = this.groupings.computeIfAbsent((group != null ? group : deferredImport),key -> new DeferredImportSelectorGrouping(createGroup(group)));//创建组grouping.add(deferredImport);//创建一个组,并加入DeferredImportSelectorHolderthis.configurationClasses.put(deferredImport.getConfigurationClass().getMetadata(),deferredImport.getConfigurationClass());//将注解属性和ConfigurationClass映射}
DeferredImportSelectorGrouping
这里就是同一个组的会添加进deferredImports
集合。
其实是比较绕的,绕了几层,我画个图看了清晰: