08、SpringBoot 源码分析 - 自动配置深度分析一

SpringBoot 源码分析 - 自动配置深度分析一

refresh和自动配置大致流程

如何自动配置

我们看下这段代码,红色框是什么放入的注册类,其实这就是我们要注册的配置类,自动装配跟他有关:

java 复制代码
public static void main(String[] args) {
		SpringApplication.run(SpringBootLearnApplication.class, args);
	}

为什么呢,因为他头上有注解 SpringBootApplication

java 复制代码
@SpringBootApplication
public class SpringBootLearnApplication

所以其实你只要传一个头上有这个注解的类就可以了,不一定要是main方法的类。

SpringBootApplication注解

那为什么就有这个注解就能自动配置呢,我们来看看这个注解,其他你可以不管,但是EnableAutoConfiguration你得看:

首先这个是一个配置类,可以被解析,因为有SpringBootConfiguration注解,被Configuration注解了:

这样我们的SpringBootApplication就类似于我们经常配置的java配置类。有了这个前提,我们就可以先看这个配置类的解析过程,其实就是ConfigurationClassPostProcessor的解析处理,解析流程我以前的spring源码文章都讲过,其中processImports的处理也特别讲过,会先递归获取所有import进来的类,然后按不同类型进行判断处理,重点就是这里啦,我们先来看下EnableAutoConfiguration注解是不是有import呢。

EnableAutoConfiguration注解

有个AutoConfigurationImportSelectorimport,其实还有一个在AutoConfigurationPackage注解里,暂时不说,重点是AutoConfigurationImportSelector

AutoConfigurationImportSelector自动配置导入选择器

看下他的结构,左边不用管,就是为了回调拿到一些属性,好对容器操作,主要是右边,我们前面spring的文章讲过ImportSelector的原理,想看的朋友可以看下这篇文章,有这个基础对下面的理解比较好,因为这里还有个DeferredImportSelector,他会在ConfigurationClassParser解析方法的最后来处理。

可以看到,他被处理的时候:

DeferredImportSelectorHandler就是专门来处理这些DeferredImportSelector的。

DeferredImportSelectorHandler的handle

首先会封装一个持有器DeferredImportSelectorHolder,如果deferredImportSelectors为空,表示在被DeferredImportSelectorGroupingHandler处理中,DeferredImportSelectorGroupingHandler又是什么呢,其实我们要处理的DeferredImportSelectorHolder会根据Group进行分组,来区分不同的ImportSelector,比如我们这个自动装配的AutoConfigurationImportSelector,是属于AutoConfigurationGroup类型的,分组处理方便,到时候只需要遍历所有组中的ImportSelector统一处理即可。因为处理的方式是迭代器循环,所以不能添加,只能直接处理,当然如果不为空,表示没有在处理,可以添加到deferredImportSelectors集合里。

java 复制代码
		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里面有集合。

java 复制代码
		//自动装配类型和组映射
		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);//创建一个组,并加入DeferredImportSelectorHolder
			this.configurationClasses.put(deferredImport.getConfigurationClass().getMetadata(),
					deferredImport.getConfigurationClass());//将注解属性和ConfigurationClass映射
		}

DeferredImportSelectorGrouping

这里就是同一个组的会添加进deferredImports集合。

其实是比较绕的,绕了几层,我画个图看了清晰:

相关推荐
计算机毕设指导6几秒前
基于 SpringBoot 的作业管理系统【附源码】
java·vue.js·spring boot·后端·mysql·spring·intellij-idea
Gu Gu Study2 分钟前
枚举与lambda表达式,枚举实现单例模式为什么是安全的,lambda表达式与函数式接口的小九九~
java·开发语言
Chris _data4 分钟前
二叉树oj题解析
java·数据结构
牙牙70510 分钟前
Centos7安装Jenkins脚本一键部署
java·servlet·jenkins
paopaokaka_luck18 分钟前
[371]基于springboot的高校实习管理系统
java·spring boot·后端
以后不吃煲仔饭30 分钟前
Java基础夯实——2.7 线程上下文切换
java·开发语言
进阶的架构师31 分钟前
2024年Java面试题及答案整理(1000+面试题附答案解析)
java·开发语言
The_Ticker36 分钟前
CFD平台如何接入实时行情源
java·大数据·数据库·人工智能·算法·区块链·软件工程
大数据编程之光1 小时前
Flink Standalone集群模式安装部署全攻略
java·大数据·开发语言·面试·flink
爪哇学长1 小时前
双指针算法详解:原理、应用场景及代码示例
java·数据结构·算法