简单详细的SpringBoot自动配置原理解析_spring


前言

上一篇文章我们介绍了​​SpringFactoriesLoader​​,之所以介绍​​SpringFactoriesLoader​​是因为我们这篇文章要介绍的SpringBoot的自动配置会用到​​SpringFactoriesLoader​​的知识。闲话少叙,让我们直入主题。

环境

spring-boot 1.5.8.RELEASE

从启动类开始

@SpringBootApplication
public class HelloworldDemoApplication {

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

如上,就是我们SpringBoot应用的启动类。让我们把眼光聚焦到​​@SpringBootApplication​​注解上面。这个注解是SpringBoot项目的主配置类。

@SpringBootApplication

//省略部分注解
@SpringBootConfiguration
@EnableAutoConfiguration
@ComponentScan(excludeFilters = {
@Filter(type = FilterType.CUSTOM, classes = TypeExcludeFilter.class),
@Filter(type = FilterType.CUSTOM, classes = AutoConfigurationExcludeFilter.class) })
public @interface SpringBootApplication {
}

根据前几篇的介绍,我们可以知道​​@SpringBootApplication​​注解是一个组合注解。​​@SpringBootConfiguration​​注解 表示这是SpringBoot的配置类,​​@ComponentScan​​ 开启组件扫描​​@EnableAutoConfiguration​​这个注解的作用就是让SpringBoot开启自动配置。自动配置的奥秘全都在这里:

@EnableAutoConfiguration

//省略部分注解
@AutoConfigurationPackage
@Import(AutoConfigurationImportSelector.class)
public @interface EnableAutoConfiguration {

}

如上我们可以看到​​EnableAutoConfiguration​​注解上有两个注解


  1. ​@AutoConfigurationPackage​​ 注解,
    从字面意思上来看就是自动配置包。点进去可以看到就是⼀个 ​​@Import​​ 注解:​​@Import(AutoConfigurationPackages.Registrar.class)​​ ,导⼊了⼀个
    Registrar 的组件,这个注解的作用就是将主配置类(​​@SpringBootConfiguration​​标注的类)所在的包及其下面所有子包里面所有的组件扫描到IOC容器中。所以说,默认情况下主配置类所在包及其子包以外的组件,Spring IOC容器是扫描不到的。

  2. ​@Import(AutoConfigurationImportSelector.class)​​通过​​@Import​​导入了​​EnableAutoConfigurationImportSelector​​类,而这个类的​​selectImports​​方法会通过​​SpringFactoriesLoader​​得到大量的配置类。而每个配置类则根据条件化配置类做出决策,以实现自动配置的功能。下面就让我们来看看​​selectImports​​方法。


EnableAutoConfigurationImportSelector的selectImports方法

 public String[] selectImports(AnnotationMetadata annotationMetadata) {
//省略部分代码
List<String> configurations = getCandidateConfigurations(annotationMetadata,
attributes);
return StringUtils.toStringArray(configurations);
}

protected List<String> getCandidateConfigurations(AnnotationMetadata metadata,
AnnotationAttributes attributes) {

List<String> configurations = SpringFactoriesLoader.loadFactoryNames(
getSpringFactoriesLoaderFactoryClass(), getBeanClassLoader());
return configurations;
}

protected Class<?> getSpringFactoriesLoaderFactoryClass() {
return EnableAutoConfiguration.class;
}

如上代码,自动配置核心的代码我都罗列出来了,最核心的就是​​loadFactoryNames​​方法,其主要有三步:

  1. 从classpath下获取所有​​META-INF/spring.factories​​这个文件下的信息。

  2. 将上面获取到的信息封装成Enumeration返回

  3. 遍历Enumeration,然后获取key为​​EnableAutoConfiguration​​下的所有值。
    ​META-INF/spring.factories​​ 这类⽂件是什么就不懵了。当然在很多第三⽅依赖中
    都会有这个⽂件,⼀般每导⼊⼀个第三⽅的依赖,除了本⾝的jar包以外,还会有⼀个 xxx-spring-boot-autoConfigure,这个就
    是第三⽅依赖⾃⼰编写的⾃动配置类。我们现在就以 spring-boot-autocongigure 这个依赖来说下,其下面的​​META-INF/spring.factories​​文件。

# Auto Configure
org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
org.springframework.boot.autoconfigure.admin.SpringApplicationAdminJmxAutoConfiguration,\
org.springframework.boot.autoconfigure.aop.AopAutoConfiguration,\
org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration,\


以DataSourceAutoConfiguration进行说明


通过上面的方式,所有的自动配置类都被导进主配置类中,但是这么多的配置类,明显有很多我们平常是没有使用到的,没必要全部生效,下面我们以​​DataSourceAutoConfiguration​​配置类为例来看一下自动配置类是如何工作的:

@Configuration
@ConditionalOnClass({ DataSource.class, EmbeddedDatabaseType.class })
@EnableConfigurationProperties(DataSourceProperties.class)
@Import({ Registrar.class, DataSourcePoolMetadataProvidersConfiguration.class })
public class DataSourceAutoConfiguration {

}

​@Configuration​​ 注解表明了​​DataSourceAutoConfiguration​​类是一个JavaConfig配置类。​​@ConditionalOnClass​​只有当classpath中存在​​DataSource​​类或者​​EmbeddedDatabaseType​​类时才启动这个配置。​​@EnableConfigurationProperties​​这个注解的作用是将将​​DataSource​​类注入到IOC容器中。​​@Import({ Registrar.class, DataSourcePoolMetadataProvidersConfiguration.class })​​是要导入额外的配置​​DataSourcePoolMetadataProvidersConfiguration​​。

DataSourceProperties 类

下面我们就来一个个看一下:首先是​​DataSourceProperties​​类:

@ConfigurationProperties(prefix = "spring.datasource")
public class DataSourceProperties
implements BeanClassLoaderAware, EnvironmentAware, InitializingBean {

private ClassLoader classLoader;

private Environment environment;

private String name = "testdb";


private boolean generateUniqueName;


private Class<? extends DataSource> type;


private String driverClassName;

private String url;

}

DataSourceProperties 通过​​@ConfigurationProperties​​注解将配置文件的前缀为(spring.datasource)的配置信息与自身的属性绑定。所有在配置⽂件中能配置的属性都是在 xxxProperties 类中封装着;配置⽂件能配置什么就可以参照某个功能对应的这个属性

类。

DataSourcePoolMetadataProvidersConfiguration 类

​DataSourcePoolMetadataProvidersConfiguration​​ 类是数据库连接池提供者的一个配置类。即classpath中存在​​org.apache.tomcat.jdbc.pool.DataSource.class​​则使⽤tomcat-jdbc连接池,如果classpath中存在 HikariDataSource.class则使⽤ Hikari连接池,如果存在​​org.apache.commons.dbcp.BasicDataSource.class​​则启用dbcp 连接池。

总结


  1. SpringBoot启动会加载大量的自动配置类。
  2. 给容器中自动配置添加组件的时候,会从​​propeties​​类中获取配置文件中指定这些属性的值。xxxAutoConfiguration:⾃动配置类给容器中添加组件。xxxProperties:封装配置⽂件中相关属性。