猜想
按照约定,从某个文件中获取所有的配置文件全路径,通过ImportSelector进行批量注册(ImportSelector没看过的看我上一篇文章)。
案例
单独一个项目书写stater
@Configuration
public class ZConfiguration {
@Bean
public ZEduCore zEduCore(){
return new ZEduCore();
}
}
public class ZEduCore {
public String study(){
System.out.println("learning stater");
return "baidu.com";
}
}
META-INF下创建spring.factories
org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
com.xxx.autoconfiguration.demo.ZConfiguration
还可以创建条件控制spring-autoconfigure-metadata.properties(可选,如果配置了,请满足相应条件)
com.gupaoedu.autoconfiguration.demo.GupaoConfiguration.ConditionalOnClass=com.xxx.DemoClass
另一个类引用其maven矩阵
@SpringBootApplication
public class SpringBootDemoApplication {
public static void main(String[] args) {
ConfigurableApplicationContext ca=SpringApplication.run(SpringBootDemoApplication.class, args);
System.out.println(ca.getBean(ZEduCore.class));
}
}
如果满足条件控制,则可以正常输出类信息。
原理
@SpringBootApplication中有一个注解@EnableAutoConfiguration
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@AutoConfigurationPackage
@Import({AutoConfigurationImportSelector.class})
public @interface EnableAutoConfiguration {
String ENABLED_OVERRIDE_PROPERTY = "spring.boot.enableautoconfiguration";
Class<?>[] exclude() default {};
String[] excludeName() default {};
}
兴奋了,和猜想相同,使用了ImportSelector进行了批量注册。
AutoConfigurationImportSelector
public String[] selectImports(AnnotationMetadata annotationMetadata) {
if (!this.isEnabled(annotationMetadata)) {
return NO_IMPORTS;
} else {
AutoConfigurationImportSelector.AutoConfigurationEntry autoConfigurationEntry = this.getAutoConfigurationEntry(annotationMetadata);
return StringUtils.toStringArray(autoConfigurationEntry.getConfigurations());
}
}
protected AutoConfigurationImportSelector.AutoConfigurationEntry getAutoConfigurationEntry(AnnotationMetadata annotationMetadata) {
if (!this.isEnabled(annotationMetadata)) {
return EMPTY_ENTRY;
} else {
AnnotationAttributes attributes = this.getAttributes(annotationMetadata);
List<String> configurations = this.getCandidateConfigurations(annotationMetadata, attributes);//返回所有的配置类全路径名
configurations = this.removeDuplicates(configurations);//
Set<String> exclusions = this.getExclusions(annotationMetadata, attributes);//需要排除的配置
this.checkExcludedClasses(configurations, exclusions);
configurations.removeAll(exclusions);
configurations = this.getConfigurationClassFilter().filter(configurations);
this.fireAutoConfigurationImportEvents(configurations, exclusions);//发送消息通知
return new AutoConfigurationImportSelector.AutoConfigurationEntry(configurations, exclusions);
}
}
AutoConfigurationImportSelector
protected List<String> getCandidateConfigurations(AnnotationMetadata metadata, AnnotationAttributes attributes) {
List<String> configurations = SpringFactoriesLoader.loadFactoryNames(this.getSpringFactoriesLoaderFactoryClass(), this.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;
}
protected Class<?> getSpringFactoriesLoaderFactoryClass() {
return EnableAutoConfiguration.class;
}
SpringFactoriesLoader
public static final String FACTORIES_RESOURCE_LOCATION = "META-INF/spring.factories";
public static List<String> loadFactoryNames(Class<?> factoryType, @Nullable ClassLoader classLoader) {
String factoryTypeName = factoryType.getName();
return loadSpringFactories(classLoader).getOrDefault(factoryTypeName, Collections.emptyList());
}
private static Map<String, List<String>> loadSpringFactories(@Nullable ClassLoader classLoader) {
MultiValueMap<String, String> result = cache.get(classLoader);
if (result != null) {
return result;
}
try {
Enumeration<URL> urls = (classLoader != null ?
classLoader.getResources(FACTORIES_RESOURCE_LOCATION) :
ClassLoader.getSystemResources(FACTORIES_RESOURCE_LOCATION));
result = new LinkedMultiValueMap<>();
while (urls.hasMoreElements()) {
URL url = urls.nextElement();
UrlResource resource = new UrlResource(url);
Properties properties = PropertiesLoaderUtils.loadProperties(resource);
for (Map.Entry<?, ?> entry : properties.entrySet()) {
String factoryTypeName = ((String) entry.getKey()).trim();
for (String factoryImplementationName : StringUtils.commaDelimitedListToStringArray((String) entry.getValue())) {
result.add(factoryTypeName, factoryImplementationName.trim());
}
}
}
cache.put(classLoader, result);
return result;
}
catch (IOException ex) {
throw new IllegalArgumentException("Unable to load factories from location [" +
FACTORIES_RESOURCE_LOCATION + "]", ex);
}
}
哦吼,从META-INF/spring.factories获取所有的配置文件,存入缓存缓存中,缓存格式Map<K, List<V>>,在根据key:EnableAutoConfiguration全路径名获取所有的配置类,通过ImportSelector批量注册