创建一个springboot项目
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>2.4.5</version>
<relativePath/> <!-- lookup parent from repository -->
</parent>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
</dependencies>
@SpringBootApplication
public class BootTestApplication {
public static void main(String[] args) {
SpringApplication.run(BootTestApplication.class,args);
}
}
@RestController
public class KissController {
@GetMapping("/kiss")
public String kiss(){
return "hello word";
}
}
访问http://localhost:8080/kiss
@SpringBootApplication
点进去@SpringBootApplication,会看见下图这几个注解
@Target(ElementType.TYPE)、@Retention(RetentionPolicy.RUNTIME) 、@Documented、@Inherited这四个是基础注解 ,@SpringBootConfiguration、@EnableAutoConfiguration、@ComponentScan这三个注解才是重要的。
@SpringBootConfiguration: Spring Boot的配置类,标注在某个类上,表示这是一个Spring Boot的配置类。
点进去@SpringBootConfiguration,会发现有一个@Configuration注解,这就是为什么是配置类了
@ComponentScan: 扫描包,在sping是有一个context:component-scan的标签来扫描类,在springboot里面为什么不用加标签扫描默认就可以扫描,就是使用了 这个@ComponentScan 注解默认扫描, @SpringBootApplication这个启动类在那个包路径下就扫描那个包的路径,excludeFilters这个属性就是排除不扫码那些类的。
@EnableAutoConfiguration: 追核心的注解,开启springboot自动配置功能,@EnableAutoConfiguration告诉SpringBoot开启自动配置,会帮我们自动去加载 自动配置类
点进去@EnableAutoConfiguration会看见@AutoConfigurationPackage、@Import({AutoConfigurationImportSelector.class})注解
@AutoConfigurationPackage: 将当前配置类所在包保存在BasePackages的Bean中。供Spring内部使用,比如springDataJpa
下面看看@AutoConfigurationPackage的源码
点进Registrar,得到一个当前配置类下的路径
进入AutoConfigurationPackages.register方法,这个方法注册了一个BasePackages的Bean
进入BasePackages类,只有一个BasePackages方法,就是用来返回配置类所在的包,提供给Spring内部使用,比如给SpringDataJPA,需要这个路径去扫码项目的pojo
@Import({AutoConfigurationImportSelector.class}),这个AutoConfigurationImportSelector类就是自动读取最核心的类了,点进去会看见实现了一个叫DeferredImportSelector接口
进入了DeferredImportSelector接口,会看见有一个getImportGroup方法
在AutoConfigurationImportSelector类里重写了getImportGroup方法,这个方法返回了一个AutoConfigurationGroup类。
进入AutoConfigurationGroup类,AutoConfigurationGroup类里有两个方法process和selectImports
process就是读取springboot第三方所有的配置类,下面断点进入看流程。先进入到process方法里的getAutoConfigurationEntry方法这一行
在断点可以看到有个23个自动配置类,比如WebMvcAutoConfiguration、MultipartAutoConfiguration这些全是springmvc的配置类,这23个全是springmvc的配置类,放开断点重新进入来看看这23个配置类是怎么的来的。
再次断点到process方法里的getAutoConfigurationEntry方法这一行
进入getAutoConfigurationEntry方法,断点到getCandidateConfigurations这一行
进入getCandidateConfigurations方法
注意这个this.getSpringFactoriesLoaderFactoryClass()方法,返回了一个org.springframework.boot.autoconfigure.EnableAutoConfiguration
进入loadFactoryNames方法
factoryTypeName就是上面那个org.springframework.boot.autoconfigure.EnableAutoConfiguration
进入loadSpringFactories方法
loadSpringFactories方法里发现result已经有值了,这是怎么回事,放开断点回到启动类BootTestApplication。
在BootTestApplication里,进入SpringApplication
在SpringApplication构造方法里有一个getSpringFactoriesInstances方法
进入getSpringFactoriesInstances方法,发现SpringFactoriesLoader.loadFactoryNames(type, classLoader)
这段代码,说明在创建SpringApplication构造器的时候已经调用了所以才可以再缓存直接拿到,来看一下这一段代码的流程。
直接断点到loadSpringFactories流程,url的返回值jar:file:/D:/java/maven/reposity/org/springframework/boot/spring-boot/2.3.6.RELEASE/spring-boot-2.3.6.RELEASE.jar!/META-INF/spring.factories,意思是在maven本地仓库去读取spring-boot-2.3.6.RELEASE.jar包下META-INF文件下的spring.factories.
打开spring.factories
在while方法里会循环出多个spring.factories拿去出里面的数据存放到result里面去
查看出result里的数据
查看打箭头这一个,末尾全是AutoConfiguration结尾
返回到loadFactoryNames这里面,调用getOrDefault进行过滤,过滤条件就是factoryTypeName
返回到getCandidateConfigurations方法,发现调用loadFactoryNames返回了127个自动配置类,刚才就是过滤出了127个自动配置类其他的不需要
返回到getAutoConfigurationEntry方法,再到this.getConfigurationClassFilter().filter(configurations)这一行,发现又过滤了23个,这23个都是springmvc和test的配置类,从127个配置类过滤出来的。
为什么是23个自动配置类,不是127个呢,因为我们的依赖只导入了两个依赖,只会从127个配置类里筛选出这两个依赖需要的配置类
将返回的23个配置类添加到autoConfigurationEntries里面去
selectImports方法,上面我们讲到过两个关键方法一个是process,一个就是selectImports,这个方法就是将自动配置类排序,也就是上面的23个,自动配置类并不是一定写死23个,而是根据整合第三方库的自动配置类和在pom文件下的依赖来筛选的