关于程序员:Spring-Boot-自动装配原理你知道吗

32次阅读

共计 3100 个字符,预计需要花费 8 分钟才能阅读完成。

前言

如果你在面试过程中被问到 Spring Boot 主动拆卸原理,你该如何答复?
没有深刻理解的小伙伴一下子就被问蒙了,这还是我意识的 Spring Boot 吗?心想说:我不会。然而对于简历上写着把握 Spring Boot 却又有点说不过去,于是进入难堪的地步。
所以为了防止上述问题,请认真浏览本篇文章,我将通过以下三个方面来解答该问题。

1、什么是 Spring Boot 主动拆卸?
2、Spring Boot 主动拆卸如何实现?
3、如何本人入手创立一个 Spring Boot Starter?

有木有感觉这三个问题有点相熟?的确,像人生的“灵魂三问”–“你是谁?”“你来自哪?”“你要去哪儿?”,
这个问题虽在哲学界是自古到今无解的难题,但在咱们学习一个新知识点时,却是要走的三步 – 这是什么?它用在哪?该怎么用?即便在有些文章中没有提及,咱们也要认真思考下这三步走,能力更深刻的学会一个新的知识点。

啥?你讲常识就讲常识,谈什么人生哲学!哼~

好了废话不多讲,带着三问,间接开整~

1、什么是 Spring Boot 主动拆卸?

Spring Boot 本人定义的一套接口标准,这套标准规定:Spring Boot 在启动时会扫描内部利用 jar 包中的 META-INF/spring.factories 文件,而后将文件中配置的类加载到 Spring 容器。
有没有感觉是 SPI 的形式,感兴趣的小伙伴能够去深刻理解,这里就不深入探讨。

2、Spring Boot 主动拆卸如何实现?

须要留神:此处应用的版本是 Spring Boot 2.7.4,不同版本之间可能源码会有所不同。

仔细的小伙伴可能会发现,咱们在创立 Spring Boot 我的项目时,在启动类上都有一个注解 @SpringBootApplication,没错咱们的问题从这里开展。

进入 @SpringBootApplication 之后,咱们把焦点放到 @EnableAutoConfiguration,它的作用就是 启动 Spring Boot 的主动配置

持续进入 @EnableAutoConfiguration 注解

能够看到,他有两个重要的注解:
@AutoConfigurationPackage:将增加该注解的类所在的 package 作为 主动配置 package 进行治理,也就说包下的所有组件主动注册到容器中
@Import({AutoConfigurationImportSelector.class}):加载主动拆卸类

咱们把焦点放到 AutoConfigurationImportSelector 类上

能够看到,该类实现了 DeferredImportSelector 接口,也就实现了外面的 selectImports 办法

isEnabled(annotationMetadata):是否启用主动配置,默认开启。
getAutoConfigurationEntry(annotationMetadata):获取所有须要主动拆卸的类

紧接着,咱们持续查看 getAutoConfigurationEntry() 办法,看它外部是如何实现的。

1、获取 @EnableAutoConfiguration 配置的 excludeexcludeName 用于排除相干类不主动拆卸
2、获取所有 Spring Boot Starter META-INF/spring.factories 文件所有主动拆卸类
而后进行一去重、排除、过滤,剩下的配置类将会被加载到 Spring 容器中。

此处可能有小伙伴会有疑难:所有配置类都会失效吗?

答案是否定的,在 过滤 里 getConfigurationClassFilter().filter(configurations),它会通过相似 @ConditionalOnXX注解去过滤出满足条件的类。就拿 @ConditionalOnMissingBean 来说,当容器里没有指定 Bean 时才会装载,也能够说是按需加载。你曾经有了,那就用你的,没有的话就用我的。

以下配上 spring.factories 文件内容

1、文件地位以及我用的版本为 2.7.4,此版本绝对于 2.7 之前的版本会有所改变。

这里援用官网版本更新形容:

Spring Boot 2.7 已不举荐应用 spring.factories

主动配置更改为:多加了一级 spring 目录,并创立红框命名文件

Spring Boot 3.0 将删除 Spring Boot 2.x 不举荐类、办法、和属性。

可能有小伙伴会有疑难,spring.factories 都快删掉了,还在跟我讲~
这里我解释下:因为 Spring Boot 2.7.x 是这个文件的过渡版本,它同时存在两个文件,起到兼顾 2.7 版本之前的小伙伴们,也能够通过比照看出其中的变动。

3、如何本人入手创立一个 Spring Boot Starter?

spring-boot-starter 目录构造:

│  pom.xml
│
├─src
│  └─main
│      ├─java
│      │  └─com
│      │      └─cpz
│      │          └─starter
│      │              ├─bean
│      │              │      MyBean.java
│      │              │
│      │              └─config
│      │                  │  BeanConfig.java
│      │                  │
│      │                  └─props
│      │                          BeanProperties.java
│      │
│      └─resources
│          └─META-INF
│                  spring.factories

外围类:BeanConfig.java

解析:
1、@EnableConfigurationProperties(BeanProperties.class):启用读取配置文件配置,其中BeanProperties 蕴含两个属性 enable 是否启用被注解类 和 name 用于测试输入不同名字。
2、@ConditionalOnClass(MyBean.class):确保 MyBean.class 存在。
3、@ConditionalOnProperty(prefix = "bean", name = "enable", matchIfMissing = true):配置文件中
bean.enable 属性为 true 时加载,缺失为 true。
4、@ConditionalOnMissingBean(MyBean.class):容器中没有 MyBean.class 时创立。

主动拆卸:spring.factories

最初,在新我的项目中应用:
pom.xml

<dependency>
    <groupId>com.cpz</groupId>
    <artifactId>spring-boot-starter</artifactId>
    <version>${revision}</version>
</dependency>

application.yml

bean:
  enable: true
  name: 我的第一个 Spring Boot Starter

IndexController.java

private final MyBean myBean;

@Operation(summary = "测试自定义 Spring Boot Starter")
@GetMapping("/bean/test")
public String beanTest() {return myBean.sayHi();
}

输入后果:

总结

Spring Boot 通过 @EnableAutoConfiguration 开启主动拆卸,通过加载文件 META-INF/spring.factories(Spring Boot 2.7 以上版本 META-INF/spring/%s.imports) 中的配置类实现主动拆卸,通过
@ConditionalOnXX 按需加载。

示例代码

「学习交换」

能够扫上面二维码,关注「我的极简博客」公众号。

始终在谋求思路的传递而非代码的 COPY

正文完
 0