淘宝一面:“讲述一下 SpringBoot 自动装配原理?”

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

我觉得我们可以从以下几个方面回答:

  1. 什么是 SpringBoot 自动装配?

  2. SpringBoot 是如何实现自动装配的?如何实现按需加载?

  3. 如何实现一个 Starter?

篇幅问题,这篇文章并没有深入,小伙伴们也可以直接使用 debug 的方式去看看 SpringBoot 自动装配部分的源代码。

前言

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

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

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

spring-servlet.xml

<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 方法即可。

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

并且,我们通过 Spring Boot 的全局配置文件 application.propertiesapplication.yml即可对项目进行设置比如更换端口号,配置 JPA 属性等等。

为什么 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.>@EnableAutoConfigurationpublic @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}) //加载自动装配类 xxxAutoconfigurationpublic @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包实现起步依赖

(0)

相关推荐

  • SpringBoot实战项目教程,真TM香!

    本章内容  Spring Boot简化Spring应用程序开发  Spring Boot的基本特性  Spring Boot工作区的设置 以下知识点解析来自于<SpringBoot实战&g ...

  • 徒手撸一个Spring Boot中的starter,解密自动化配置

    starter背景 Spring Boot目前已经变成了后端开发这必备技能之一,其中一个主要原因是Spring Boot中有个非常重要的机制(starter机制). starter能够抛弃以前繁杂的配 ...

  • SpringBoot源码解析(一)

    不知道大家在第一次搭建SpringBoot项目时是否有注意到,没有繁琐的配置文件,引入依赖时也不需要指定版本.整合第三方框架,如果需要开启相应的功能,只需要在启动类上添加@Enablexxxx注解即可 ...

  • SpringBoot自动配置原理

    我们知道,SpringBoot项目创建完成后,即使不进行任何的配置,也能够顺利地运行,这都要归功于SpringBoot的自动化配置. SpringBoot默认使用application.propert ...

  • SpringBoot-常见问题(一)

    参考: https://www.cnblogs.com/nihaorz/p/10528121.html https://www.cnblogs.com/tanwei81/p/6814022.html ...

  • SpringBoot运行原理

    pom.xml <parent> <groupId>org.springframework.boot</groupId> <artifactId>spr ...

  • SpringBoot面试杀手锏——自动配置原理

    引言 不论在工作中,亦或是求职面试,Spring Boot已经成为我们必知必会的技能项.除了某些老旧的政府项目或金融项目持有观望态度外,如今的各行各业都在飞速的拥抱这个已经不是很新的Spring启动框 ...

  • SpringBoot 注解原理,自动装配原理,图文并茂,万字长文!

    首先,先看SpringBoot的主配置类: @SpringBootApplication public class StartEurekaApplication {    public static ...

  • 淘宝拒收以后会自动退款吗

    很多用户在淘宝上购物后,发现自己的包裹有破损,或者是根本没收到物品,那么淘宝拒收以后会自动退款吗,淘宝拒收以后会自动退款,但是最好和店家说明情况,是商品包裹有损坏还是没有收到商品,这样可以加快退款速度 ...

  • 解密淘宝自动静默关注功能投放钻展 超级推荐

    这个是完全无提示的情况下,让你不知不觉的关注了店铺 我们都知道,淘宝现在主推的是内容化营销,这种营销口号的推出,使得淘宝店铺粉丝的价值变得更加重要.我们在浏览店铺的时候,如果喜欢这个店铺,可以主动关注 ...

  • Store精选案例丨淘宝自动插旗&备注机器人

    UB Store精选案例 点击阅读原文,立即定制! 淘宝自动插旗&备注机器人 档案 机器人简介 能够自动给淘宝订单添加旗子类型与备注的电商运营机器人. ◆支持自动给订单添加旗子类型 ◆支持自动 ...

  • 淘宝千牛商品管理软件怎么设置自动上下架商品

    淘宝卖家们使用千牛商品管理软件可以让工作轻松很多,其实千牛还可以自动上下架宝贝.下面小编就来教大家淘宝千牛商品管理软件怎么设置自动上下架商品. 淘宝千牛商品管理软件怎么设置自动上下架商品 先登录千牛工 ...

  • 【筋脉石大家谈】第23期 老马来讲述《心里的历程&#183;淘宝在路上》

    集宝街--玉见魅力石界 Youth Day " 石友们,大家好![筋脉石大家谈]已经到来到了第23期,本期"筋脉石爱好者乐园"诚邀到了筋脉石的资深玩家--马润河老师,由他 ...

  • 解放耍弄收,双十二自动刷淘宝能量,这个脚本你值得拥有

    重磅干货,第一时间送达 双十一的余温还没有过去,每天还沉浸在收快递的忙碌之中,悄悄地,双十二又来了. 不知道这几天,双十二的收集能量活动有没有让各位小伙伴厌烦?但是为了能便宜那0.01元钱,还是得咬牙 ...

  • 双十二自动刷淘宝能量,推荐这个厉害的脚本!

    这几天,双十二的收集能量活动有没有把你折腾的不轻呢?接下来给大家介绍一个脚本可自动收集能量的,彻底解放双手! 接下来,教大家怎么玩? 1.在手机上安装Auto.js.(软件可以在公众号上回复:脚本 . ...