SpringBoot中spring.factories指定的EnableAutoConfiguration并不是按照配置顺序执行 ! ! !
本文基于SpringBoot 2.6.4分析
先看示例:
@Configuration
public class DbAutoConfig {
public DbAutoConfig() { System.out.println("DbAutoConfig"); }
@Bean
public Bean1 dbBean() {
System.out.println("db bean");
return new Bean1();
}
}
@Configuration
public class RedisAutoConfig {
public RedisAutoConfig() { System.out.println("RedisAutoConfig"); }
@Bean
public Bean1 redisBean() {
System.out.println("redis bean");
return new Bean1();
}
}
在META-INF/spring.factories中配置如下:

按照一开始的设想:由于RedisAutoConfig配置在DbAutoConfig的前面,所以redisBean应该在dbBean之前被创建 ?
然而这个设想是错误的,dbBean先被创建了,输出结果如下。

源码分析:
开始o.s.core.io.support.SpringFactoriesLoader#loadFactoryNames确实是从spring.factories按照顺序加载配置类保存在List中,但是在执行o.s.boot.a.AutoConfigurationImportSelector.AutoConfigurationGroup#selectImports()时会对这些配置类进行排序后再注册到Spring上下中

排序规则如下:

- 先按照配置类的全类名字符串进行排序
- 再按照配置类上的
@AutoConfigureOrder注解排序 - 最后按照
@AutoConfigureBefore@AutoConfigureAfter两个注解排序
如上就是示例中为什么配置类不是按照在spring.factories配置的顺序执行的原因!!!
针对这个问题,曾在2020年的时候给SpringBoot提过,希望按照在spring.factories配置的顺序执行,但是被否决了,不过wilkinsona的回复也是有一定道理的。所以我们知道就好 ! ! !
It is suggested that the key of EnableAutoConfiguration in spring.factories can be sorted

@AutoConfigureOrder @AutoConfigureBefore @AutoConfigureAfter三个注解元数据是怎么读取的?
每个自动装配的配置类都会被封装在
org.springframework.boot.autoconfigure.AutoConfigurationSorter.AutoConfigurationClass中,获取注解元数据时通过org.springframework.core.type.classreading.MetadataReaderFactory#getMetadataReader(java.lang.String className)获取MetadataReader来读取注解元数据,MetadataReader实际上内部又使用ASM框架读取字节码来实现。

spring.factories中同一个key配置的value有重复时,是否会导致多个实例对象
当前看的这个版本org.springframework.core.io.support.SpringFactoriesLoader在加载配置类时会去重,不会重复。但是低版本可能出现多个实例对象。
请看spring-framework的修复记录:Revise duplicate filtering in SpringFactoriesLoader











