SpringBoot自动配置原理(源码)

学习SpringBoot源码需要了解Spring相关源码,这样才能更通透。

一、原理简述

SpringBoot启动类本身也是一个配置类,Spring会解析配置类上的所有嵌套的@Import注解,其中就有@EnableAutoConfiguration注解中的@Import(AutoConfigurationImportSelector.class),AutoConfigurationImportSelector类就是自动配置的关键,他的主要作用是获取META-INF/spring.factories文件中以EnableAutoConfiguration为key的值,这些值全是可能需要的自动配置类。例如:
在这里插入图片描述
获取到这些自动配置类后,根据它自身的@ConditionalOnClass等条件判断是否需要导入。最后处理这些过滤后的配置类完成自动配置。

需要注意的是META-INF/spring.factories中的值在SpringBoot启动的时候就被缓存起来了,处理AutoConfigurationImportSelector的时候是从缓存中取得值,不过两者都调用的是同一个方法。

二、源码探析

@SpringBootApplication
public class MySpringBootApplication {

	public static void main(String[] args) { 
		SpringApplication.run(MySpringBootApplication.class, args);
	}
}

这是一个初始的SpringBoot项目的启动类,这个类被@SpringBootApplication注解修饰着,这是一个什么注解呢?

点进去瞅瞅。

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@SpringBootConfiguration
@EnableAutoConfiguration
@ComponentScan(excludeFilters = { @Filter(type = FilterType.CUSTOM, classes = TypeExcludeFilter.class),
		@Filter(type = FilterType.CUSTOM, classes = AutoConfigurationExcludeFilter.class) })
public @interface SpringBootApplication {

这个注解又被很多个注解修饰着,比如@SpringBootConfiguration表示这是一个配置类,也就是说咱们的启动类也是一个配置类。这些注解中,最重要的是 @EnableAutoConfiguration 注解。

这个注解的名字翻译过来就是开启自动配置,通过名字来看肯定和自动配置有关,那为什么加了这个注解就能开启自动配置呢?我们再点进去看看具体长什么样子。

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@AutoConfigurationPackage
@Import(AutoConfigurationImportSelector.class)
public @interface EnableAutoConfiguration {

我们发现这个注解上用@Import导入了一个AutoConfigurationImportSelector类,这个类翻译过来是自动注解导入选择器。自动配置的实现关键就在此了。

这里简述一下@Import注解的用处

  1. 导入@Configuration配置类,会直接解析这个配置类
  2. 可以导入@Component类
  3. ImportSelector接口实现类,Spring会执行selectImports方法,用于导入一组的bean或配置类的全名。
  4. DeferredImportSelector接口实现类,会在项目中所有配置类解析完毕后,执行Group.process方法处理导入@Configuration类。
  5. ImportBeanDefinitionRegistrar接口实现类,执行registerBeanDefinitions方法注册BeanDefintion。

AutoConfigurationImportSelector类实现了DeferredImportSelector接口,在spring容器创建中会执行它的process方法导入配置类。

在这里插入图片描述

process方法中有个getAutoConfigurationEntry方法,翻译过来是获取自动配置类条目。

在这里插入图片描述
getAutoConfigurationEntry方法中获取候选配置类,并过滤和排除一些用不着的配置类。

	protected AutoConfigurationEntry getAutoConfigurationEntry(AnnotationMetadata annotationMetadata) {
		if (!isEnabled(annotationMetadata)) {
			return EMPTY_ENTRY;
		}
		AnnotationAttributes attributes = getAttributes(annotationMetadata);
		// 获取候选配置类
		List<String> configurations = getCandidateConfigurations(annotationMetadata, attributes);
		// 过滤出重复的
		configurations = removeDuplicates(configurations);
		Set<String> exclusions = getExclusions(annotationMetadata, attributes);
		// 检查排除类
		checkExcludedClasses(configurations, exclusions);
		// 排除
		configurations.removeAll(exclusions);
		// 根据一下@ConditionalOnClass等条件注解过滤配置类
		configurations = getConfigurationClassFilter().filter(configurations);
		fireAutoConfigurationImportEvents(configurations, exclusions);
		return new AutoConfigurationEntry(configurations, exclusions);
	}

其中重要的是getCandidateConfigurations方法。

我们从下面的报错信息中也能了解到,它是从META-INF/spring.factories文件中获取自动配置类的信息。

SpringFactoriesLoader.loadFactoryNames 就是从META-INF/spring.factories文件中获取自动配置类的信息的,而key getSpringFactoriesLoaderFactoryClass() 就是org.springframework.boot.autoconfigure.EnableAutoConfiguration。

	protected List<String> getCandidateConfigurations(AnnotationMetadata metadata, AnnotationAttributes attributes) {
		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;
	}

在这里插入图片描述

获取到这些配置类之后,Spring就会处理这些配置类完成一些自动配置的工作。

我们在配置文件中配置的项,这些配置类是怎么获取到的呢?

以我们熟悉的JdbcTemplateAutoConfiguration为例。

这个自动配置类上有一个@EnableConfigurationProperties(JdbcProperties.class)注解,这个注解是用来让@ConfigurationProperties注解生效,并往容器注入一个JdbcProperties的bean。这样配置类就能从容器中获取这个bean,从而拿到咱们配置的项。

而@ConfigurationProperties注解就是将配置中以特定字符串开头的配置,映射到当前类的属性上。

@ConfigurationProperties(prefix = "spring.jdbc")
public class JdbcProperties {

@Configuration(proxyBeanMethods = false)
@ConditionalOnClass({ DataSource.class, JdbcTemplate.class })
@ConditionalOnSingleCandidate(DataSource.class)
@AutoConfigureAfter(DataSourceAutoConfiguration.class)
@EnableConfigurationProperties(JdbcProperties.class)
@Import({ JdbcTemplateConfiguration.class, NamedParameterJdbcTemplateConfiguration.class })
public class JdbcTemplateAutoConfiguration {

}

总结

SpringBoot启动时会加载META-INF/spring.factories中配置的自动配置类,并缓存起来。

Spring解析启动类上的所有嵌套的@Import注解,其中就有@EnableAutoConfiguration注解中的@Import(AutoConfigurationImportSelector.class),AutoConfigurationImportSelector类会获取META-INF/spring.factories中(从缓存中)以org.springframework.boot.autoconfigure.EnableAutoConfiguration为key的所有自动配置类,将这些自动配置类过滤排除之后,解析处理这些配置类,这些配置类用@EnableConfigurationProperties(JdbcProperties.class)的方式获取配置文件中的配置项,完成自动配置的功能。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

吖土豆

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值