本文结论
- 源码使用spring boot2.6.6版本
- 开始自动配置的核心注解:@EnableAutoConfiguration
- @EnableAutoConfiguration中使用了@Import(AutoConfigurationImportSelector.class)。
- AutoConfigurationImportSelector实现了DeferredImportSelector这个接口,Spring容器在启动时,会在解析完其他所有程序员定义的配置类之后,来调用AutoConfigurationImportSelector中的selectImports方法,然后把该方法返回的类名对应的类作为配置类进行解析。
- selectImports方法中通过spi的方式,找到所有的自动配置类
spring boot开启方式
- 一般使用@SpringBootApplication这个注解!
@SpringBootApplication
public class MyApplication {
public static void main(String[] args) {
SpringApplication.run(MyApplication.class);
}
}
@SpringBootApplication注解详细信息
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
// 下面这三个是最核心的!可以认为@SpringBootApplication是一个三合一注解。直接写这三个注解放在核心启动类上也是可以使用的!
@SpringBootConfiguration // 标记当前是一个配置类
@EnableAutoConfiguration // 开启自动配置
// 扫描
@ComponentScan(excludeFilters = {
@Filter(type = FilterType.CUSTOM, classes = TypeExcludeFilter.class),
@Filter(type = FilterType.CUSTOM,
classes = AutoConfigurationExcludeFilter.class) })
public @interface SpringBootApplication {
......
}
开启自动配置的核心注解:@EnableAutoConfiguration
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
// 将主程序类所在包及所有子包下的组件到扫描到 spring 容器中。
@AutoConfigurationPackage
// 最核心的,自动配置的处理类!
@Import(AutoConfigurationImportSelector.class)
public @interface EnableAutoConfiguration {
......
}
- @Import(AutoConfigurationImportSelector.class),Import了自动配置的核心处理类
自动配置处理类:AutoConfigurationImportSelector
- AutoConfigurationImportSelector实现了DeferredImportSelector这个接口,Spring容器在启动时,会在解析完其他所有程序员定义的配置类之后,来调用AutoConfigurationImportSelector中的selectImports方法,然后把该方法返回的类名对应的类作为配置类进行解析。
- AutoConfigurationImportSelector中的selectImports会利用SpringFactoriesLoader找到所有的META-INF/spring.factories文件中key为EnableAutoConfiguration.class的value值,也就是众多自动配置类的类名。
// 实现了DeferredImportSelector这个接口!
public class AutoConfigurationImportSelector
implements DeferredImportSelector, BeanClassLoaderAware, ResourceLoaderAware,
BeanFactoryAware, EnvironmentAware, Ordered {
......
// 会在@Configuration都解析完成后执行(解析完程序员自定义的类之后执行)
// annotationMetadata这个对象是我们main方法的信息
@Override
public String[] selectImports(AnnotationMetadata annotationMetadata) {
// 判断是否开启自动配置:spring.boot.enableautoconfiguration配置的值为true(默认值为true)。没有开启的时候返回一个空数组。
if (!isEnabled(annotationMetadata)) {
return NO_IMPORTS;
}
// 获取自动配置类(spring.factories中导入的)
AutoConfigurationEntry autoConfigurationEntry = getAutoConfigurationEntry(annotationMetadata);
// 返回配置类的类名
return StringUtils.toStringArray(autoConfigurationEntry.getConfigurations());
}
......
/**
* 获取自动配置类(spring.factories中导入的)
*/
protected AutoConfigurationEntry getAutoConfigurationEntry(AnnotationMetadata annotationMetadata) {
// 判断是否开启自动配置:spring.boot.enableautoconfiguration配置的值为true(默认值为true)。没有开启的时候返回一个空对象。
if (!isEnabled(annotationMetadata)) {
return EMPTY_ENTRY;
}
// 获取@EnableAutoConfiguration的属性值(exclude,excludeName)。作用是获取需要排除的自动配置类。
AnnotationAttributes attributes = getAttributes(annotationMetadata);
// 获取spring.factories文件中所有的org.springframework.boot.autoconfigure.EnableAutoConfiguration的值
List<String> configurations = getCandidateConfigurations(annotationMetadata, attributes);
// 按照类名去重。每个jar包都可能有spring.factories中进行了配置,内容一致进行去重。
// 去重代码为:return new ArrayList<>(new LinkedHashSet<>(list));
configurations = removeDuplicates(configurations);
// 获取需要排除的自动配置类。通过@EnableAutoConfiguration中的俩个属性(exclude,excludeName)进行指定。也可以通过属性spring.autoconfigure.exclude进行配置(逗号分隔)。
Set<String> exclusions = getExclusions(annotationMetadata, attributes);
// 排除操作
checkExcludedClasses(configurations, exclusions);
configurations.removeAll(exclusions);
// 获取spring.factories中的AutoConfigurationImportFilter进行过滤
// 拿到条件注解过滤:OnBeanCondition、OnClassCondition、OnWebApplicationCondition
// 使用spring-autoconfigure-metadata.properties(编译的时候生成的文件!里面存一下条件注解对应关系)中的配置过滤。核心作用是加快启动速度!
configurations = getConfigurationClassFilter().filter(configurations);
// 发布一个事件,可以进行日志的打印(日志级别需要trans),展示出那些自动配置了,那些没有自动配置,因为什么导致无法进行自动配置...
fireAutoConfigurationImportEvents(configurations, exclusions);
// 最终的返回,全部是合格的自动配置类!
return new AutoConfigurationEntry(configurations, exclusions);
}
/**
* 获取spring.factories文件中所有的org.springframework.boot.autoconfigure.EnableAutoConfiguration的值
*/
protected List<String> getCandidateConfigurations(AnnotationMetadata metadata, AnnotationAttributes attributes) {
// spi的方式找到spring.factories文件中所有的org.springframework.boot.autoconfigure.EnableAutoConfiguration的值
// getSpringFactoriesLoaderFactoryClass()的逻辑是return EnableAutoConfiguration.class;
List<String> configurations = SpringFactoriesLoader.loadFactoryNames(getSpringFactoriesLoaderFactoryClass(), getBeanClassLoader());
// 无值得时候,抛出异常日志
Assert.notEmpty(configurations, "No auto configuration classes found in META-INF/spring.factories. If you are using a custom packaging, make sure that file is correct.");
// 返回这些所有的自动配置类
return configurations;
}
}
结束语
- 你的点赞是我提高文章质量最大的动力!!!
- 获取更多本文的前置知识文章,以及新的有价值的文章,让我们一起成为架构师!
- 目前已经完成了并发编程、MySQL、spring源码、Mybatis的源码。可以在公众号下方菜单点击查看之前的文章!
- 这个公众号的所有技术点,会分析的很深入!
- 这个公众号,无广告!!!