聊聊 SpringBoot 自动装配原理

本文已经收录进 Github 95k+ Star 的Java项目JavaGuide 。JavaGuide项目地址 : github.com/Snailclimb/…前端

做者:Miki-byte-1024 & Snailclimbjava

每次问到 Spring Boot, 面试官很是喜欢问这个问题:“讲述一下 SpringBoot 自动装配原理?”。git

我以为咱们能够从如下几个方面回答:github

  1. 什么是 SpringBoot 自动装配?
  2. SpringBoot 是如何实现自动装配的?如何实现按需加载?
  3. 如何实现一个 Starter?

篇幅问题,这篇文章并无深刻,小伙伴们也能够直接使用 debug 的方式去看看 SpringBoot 自动装配部分的源代码。web

前言

使用过 Spring 的小伙伴,必定有被 XML 配置统治的恐惧。即便 Spring 后面引入了基于注解的配置,咱们在开启某些 Spring 特性或者引入第三方依赖的时候,仍是须要用 XML 或 Java 进行显式配置。面试

举个例子。没有 Spring Boot 的时候,咱们写一个 RestFul Web 服务,还首先须要进行以下配置。redis

@Configuration
public class RESTConfiguration {
    @Bean
    public View jsonTemplate() {
        MappingJackson2JsonView view = new MappingJackson2JsonView();
        view.setPrettyPrint(true);
        return view;
    }

    @Bean
    public ViewResolver viewResolver() {
        return new BeanNameViewResolver();
    }
}
复制代码

spring-servlet.xmlspring

<beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context" xmlns:mvc="http://www.springframework.org/schema/mvc" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context/ http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/mvc/ http://www.springframework.org/schema/mvc/spring-mvc.xsd">

    <context:component-scan base-package="com.howtodoinjava.demo" />
    <mvc:annotation-driven />

    <!-- JSON Support -->
    <bean name="viewResolver" class="org.springframework.web.servlet.view.BeanNameViewResolver"/>
    <bean name="jsonTemplate" class="org.springframework.web.servlet.view.json.MappingJackson2JsonView"/>

</beans>
复制代码

可是,Spring Boot 项目,咱们只须要添加相关依赖,无需配置,经过启动下面的 main 方法便可。数据库

@SpringBootApplication
public class DemoApplication {
    public static void main(String[] args) {
        SpringApplication.run(DemoApplication.class, args);
    }
}
复制代码

而且,咱们经过 Spring Boot 的全局配置文件 application.propertiesapplication.yml便可对项目进行设置好比更换端口号,配置 JPA 属性等等。json

为何 Spring Boot 使用起来这么酸爽呢? 这得益于其自动装配。自动装配能够说是 Spring Boot 的核心,那究竟什么是自动装配呢?

什么是 SpringBoot 自动装配?

咱们如今提到自动装配的时候,通常会和 Spring Boot 联系在一块儿。可是,实际上 Spring Framework 早就实现了这个功能。Spring Boot 只是在其基础上,经过 SPI 的方式,作了进一步优化。

SpringBoot 定义了一套接口规范,这套规范规定:SpringBoot 在启动时会扫描外部引用 jar 包中的META-INF/spring.factories文件,将文件中配置的类型信息加载到 Spring 容器(此处涉及到 JVM 类加载机制与 Spring 的容器知识),并执行类中定义的各类操做。对于外部 jar 来讲,只须要按照 SpringBoot 定义的标准,就能将本身的功能装置进 SpringBoot。

没有 Spring Boot 的状况下,若是咱们须要引入第三方依赖,须要手动配置,很是麻烦。可是,Spring Boot 中,咱们直接引入一个 starter 便可。好比你想要在项目中使用 redis 的话,直接在项目中引入对应的 starter 便可。

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
复制代码

引入 starter 以后,咱们经过少许注解和一些简单的配置就能使用第三方组件提供的功能了。

在我看来,自动装配能够简单理解为:经过注解或者一些简单的配置就能在 Spring Boot 的帮助下实现某块功能。

SpringBoot 是如何实现自动装配的?

咱们先看一下 SpringBoot 的核心注解 SpringBootApplication

@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
<1.>@SpringBootConfiguration
<2.>@ComponentScan
<3.>@EnableAutoConfiguration
public @interface SpringBootApplication {

}

@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Configuration //实际上它也是一个配置类
public @interface SpringBootConfiguration {
}
复制代码

大概能够把 @SpringBootApplication看做是 @Configuration@EnableAutoConfiguration@ComponentScan 注解的集合。根据 SpringBoot 官网,这三个注解的做用分别是:

  • @EnableAutoConfiguration:启用 SpringBoot 的自动配置机制

  • @Configuration:容许在上下文中注册额外的 bean 或导入其余配置类

  • @ComponentScan: 扫描被@Component (@Service,@Controller)注解的 bean,注解默认会扫描启动类所在的包下全部的类 ,能够自定义不扫描某些 bean。以下图所示,容器中将排除TypeExcludeFilterAutoConfigurationExcludeFilter

@EnableAutoConfiguration 是实现自动装配的重要注解,咱们以这个注解入手。

@EnableAutoConfiguration:实现自动装配的核心注解

EnableAutoConfiguration 只是一个简单地注解,自动装配核心功能的实现实际是经过 AutoConfigurationImportSelector类。

@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@AutoConfigurationPackage //做用:将main包下的所欲组件注册到容器中
@Import({AutoConfigurationImportSelector.class}) //加载自动装配类 xxxAutoconfiguration
public @interface EnableAutoConfiguration {
    String ENABLED_OVERRIDE_PROPERTY = "spring.boot.enableautoconfiguration";

    Class<?>[] exclude() default {};

    String[] excludeName() default {};
}
复制代码

咱们如今重点分析下AutoConfigurationImportSelector 类到底作了什么?

AutoConfigurationImportSelector:加载自动装配类

AutoConfigurationImportSelector类的继承体系以下:

public class AutoConfigurationImportSelector implements DeferredImportSelector, BeanClassLoaderAware, ResourceLoaderAware, BeanFactoryAware, EnvironmentAware, Ordered {

}

public interface DeferredImportSelector extends ImportSelector {

}

public interface ImportSelector {
    String[] selectImports(AnnotationMetadata var1);
}
复制代码

能够看出,AutoConfigurationImportSelector 类实现了 ImportSelector接口,也就实现了这个接口中的 selectImports方法,该方法主要用于获取全部符合条件的类的全限定类名,这些类须要被加载到 IoC 容器中

private static final String[] NO_IMPORTS = new String[0];

public String[] selectImports(AnnotationMetadata annotationMetadata) {
        // <1>.判断自动装配开关是否打开
        if (!this.isEnabled(annotationMetadata)) {
            return NO_IMPORTS;
        } else {
          //<2>.获取全部须要装配的bean
            AutoConfigurationMetadata autoConfigurationMetadata = AutoConfigurationMetadataLoader.loadMetadata(this.beanClassLoader);
            AutoConfigurationImportSelector.AutoConfigurationEntry autoConfigurationEntry = this.getAutoConfigurationEntry(autoConfigurationMetadata, annotationMetadata);
            return StringUtils.toStringArray(autoConfigurationEntry.getConfigurations());
        }
    }
复制代码

这里咱们须要重点关注一下getAutoConfigurationEntry()方法,这个方法主要负责加载自动配置类的。

该方法调用链以下:

如今咱们结合getAutoConfigurationEntry()的源码来详细分析一下:

private static final AutoConfigurationEntry EMPTY_ENTRY = new AutoConfigurationEntry();

AutoConfigurationEntry getAutoConfigurationEntry(AutoConfigurationMetadata autoConfigurationMetadata, AnnotationMetadata annotationMetadata) {
        //<1>.
        if (!this.isEnabled(annotationMetadata)) {
            return EMPTY_ENTRY;
        } else {
            //<2>.
            AnnotationAttributes attributes = this.getAttributes(annotationMetadata);
            //<3>.
            List<String> configurations = this.getCandidateConfigurations(annotationMetadata, attributes);
            //<4>.
            configurations = this.removeDuplicates(configurations);
            Set<String> exclusions = this.getExclusions(annotationMetadata, attributes);
            this.checkExcludedClasses(configurations, exclusions);
            configurations.removeAll(exclusions);
            configurations = this.filter(configurations, autoConfigurationMetadata);
            this.fireAutoConfigurationImportEvents(configurations, exclusions);
            return new AutoConfigurationImportSelector.AutoConfigurationEntry(configurations, exclusions);
        }
    }
复制代码

第 1 步:

判断自动装配开关是否打开。默认spring.boot.enableautoconfiguration=true,可在 application.propertiesapplication.yml 中设置

第 2 步

用于获取EnableAutoConfiguration注解中的 excludeexcludeName

第 3 步

获取须要自动装配的全部配置类,读取META-INF/spring.factories

spring-boot/spring-boot-project/spring-boot-autoconfigure/src/main/resources/META-INF/spring.factories
复制代码

从下图能够看到这个文件的配置内容都被咱们读取到了。XXXAutoConfiguration的做用就是按需加载组件。

不光是这个依赖下的META-INF/spring.factories被读取到,全部 Spring Boot Starter 下的META-INF/spring.factories都会被读取到。

因此,你能够清楚滴看到, druid 数据库链接池的 Spring Boot Starter 就建立了META-INF/spring.factories文件。

若是,咱们本身要建立一个 Spring Boot Starter,这一步是必不可少的。

第 4 步

到这里可能面试官会问你:“spring.factories中这么多配置,每次启动都要所有加载么?”。

很明显,这是不现实的。咱们 debug 到后面你会发现,configurations 的值变小了。

由于,这一步有经历了一遍筛选,@ConditionalOnXXX 中的全部条件都知足,该类才会生效。

@Configuration
// 检查相关的类:RabbitTemplate 和 Channel是否存在
// 存在才会加载
@ConditionalOnClass({ RabbitTemplate.class, Channel.class })
@EnableConfigurationProperties(RabbitProperties.class)
@Import(RabbitAnnotationDrivenConfiguration.class)
public class RabbitAutoConfiguration {
}
复制代码

有兴趣的童鞋能够详细了解下 Spring Boot 提供的条件注解

  • @ConditionalOnBean:当容器里有指定 Bean 的条件下
  • @ConditionalOnMissingBean:当容器里没有指定 Bean 的状况下
  • @ConditionalOnSingleCandidate:当指定 Bean 在容器中只有一个,或者虽然有多个可是指定首选 Bean
  • @ConditionalOnClass:当类路径下有指定类的条件下
  • @ConditionalOnMissingClass:当类路径下没有指定类的条件下
  • @ConditionalOnProperty:指定的属性是否有指定的值
  • @ConditionalOnResource:类路径是否有指定的值
  • @ConditionalOnExpression:基于 SpEL 表达式做为判断条件
  • @ConditionalOnJava:基于 Java 版本做为判断条件
  • @ConditionalOnJndi:在 JNDI 存在的条件下差在指定的位置
  • @ConditionalOnNotWebApplication:当前项目不是 Web 项目的条件下
  • @ConditionalOnWebApplication:当前项目是 Web 项 目的条件下

如何实现一个 Starter

光说不练假把式,如今就来撸一个 starter,实现自定义线程池

第一步,建立threadpool-spring-boot-starter工程

第二步,引入 Spring Boot 相关依赖

第三步,建立ThreadPoolAutoConfiguration

第四步,在threadpool-spring-boot-starter工程的 resources 包下建立META-INF/spring.factories文件

最后新建工程引入threadpool-spring-boot-starter

测试经过!!!

总结

Spring Boot 经过@EnableAutoConfiguration开启自动装配,经过 SpringFactoriesLoader 最终加载META-INF/spring.factories中的自动配置类实现自动装配,自动配置类其实就是经过@Conditional按需加载的配置类,想要其生效必须引入spring-boot-starter-xxx包实现起步依赖

文章结尾

我是 Guide 哥,一 Java 后端开发,会一点前端,自由的少年。咱们下期再见!微信搜“JavaGuide”回复“面试突击”领取我整理的 4 本原创PDF

相关文章
相关标签/搜索