SpringBoot 自动配置是如何实现的
2022-11-29 14:00:39 18 举报
AI智能生成
根据 JavaGuide 绘制的脑图
作者其他创作
大纲/内容
什么是 SpringBoot 自动装配?
我们现在提到自动装配的时候,一般会和 SpringBoot 联系在一起。但是,实际上 Spring Framework 早就实现了这个功能。<br>Spring Boot 只是在其基础上,通过 SPI 的方式,做了进一步优化。
SpringBoot 定义了一套接口规范,这套规范规定:SpringBoot 在启动时会扫描外部引用 jar 包中的 META-INF/spring.factories 文件,<br>将文件中配置的类型信息加载到 Spring 容器(此处涉及到 JVM 类加载机制与 Spring 的容器知识),并执行类中定义的各种操作。<br>对于外部 jar 来说,只需要按照 SpringBoot 定义的标准,就能将自己的功能装置进 SpringBoot。
没有 SpringBoot 的情况下,如果我们需要引入第三方依赖,需要手动配置,非常麻烦。但是,SpringBoot 中,<br>我们直接引入一个 starter 即可。比如你想要在项目中使用 redis 的话,直接在项目中引入对应的 starter 即可。
引入 starter 之后,我们通过少量注解和一些简单的配置就能使用第三方组件提供的功能了。
自动装配可以简单理解为:<b><font color="#0000ff">通过注解或者一些简单的配置就能在 SpringBoot 的帮助下实现某块功能。</font></b>
SpringBoot 是如何实现自动装配的?<br>
核心注解 @SpringBootApplication
我们先看一下 SpringBoot 的核心注解 @SpringBootApplication :
大概可以把 @SpringBootApplication 看作是 @Configuration、@EnableAutoConfiguration、<br>@ComponentScan 注解的集合。根据 SpringBoot 官网,这三个注解的作用分别是:<br><br><ul><li>@EnableAutoConfiguration:启用 SpringBoot 的自动配置机制</li><li>@Configuration:允许在上下文中注册额外的 bean 或导入其他配置类</li><li>@ComponentScan: 扫描被 @Component (@Service,@Controller) 注解的 bean,<br>注解默认会扫描启动类所在的包下所有的类 ,可以自定义不扫描某些 bean。如下图所示,<br>容器中将排除 TypeExcludeFilter 和 AutoConfigurationExcludeFilter。</li></ul>
@EnableAutoConfiguration 是实现自动装配的重要注解,我们以这个注解入手。<br>
@EnableAutoConfiguration:实现自动装配的核心注解<br>
@EnableAutoConfiguration 只是一个简单地注解,自动装配核心功能的实现实际是通过 AutoConfigurationImportSelector 类。
下面重点分析下 AutoConfigurationImportSelector 类到底做了什么?<br>
AutoConfigurationImportSelector:加载自动装配类<br>
AutoConfigurationImportSelector 类的继承体系如下:
可以看出,AutoConfigurationImportSelector 类实现了 ImportSelector 接口,也就实现了这个接口中的 selectImports 方法,<br>该方法主要用于 <b><font color="#0000ff">获取所有符合条件的类的全限定类名,这些类需要被加载到 IoC 容器中</font></b>。
这里我们需要重点关注一下 getAutoConfigurationEntry() 方法,这个方法主要负责加载自动配置类。
现在我们结合 getAutoConfigurationEntry() 的源码来详细分析一下:
<b><font color="#0000ff">第 1 步</font></b><font color="#0000ff"><b>:</b></font><br><ul><li><span style="font-size: inherit;">判断自动装配开关是否打开。</span></li><li><span style="font-size: inherit;">默认spring.boot.enableautoconfiguration=true,可在 application.properties 或 application.yml 中设置</span></li></ul>
<font color="#0000ff"><b>第 2 步 :</b></font><br><ul><li>用于获取 EnableAutoConfiguration 注解中的 exclude 和 excludeName。</li></ul>
<b><font color="#0000ff">第 3 步:</font></b><br><ul><li>获取需要自动装配的所有配置类,读取 META-INF/spring.factories</li></ul>
从下图可以看到这个文件的配置内容都被我们读取到了。XXXAutoConfiguration 的作用就是按需加载组件。
不光是这个依赖下的 META-INF/spring.factories 被读取到,所有 Spring Boot Starter 下的 META-INF/spring.factories 都会被读取到。<br>所以,你可以清楚滴看到, druid 数据库连接池的 Spring Boot Starter 就创建了 META-INF/spring.factories 文件。<br>
<b><font color="#0000ff">第 4 步 :</font></b><br><ul><li>到这里可能面试官会问你:“spring.factories 中这么多配置,每次启动都要全部加载么?”。</li><li>很明显,这是不现实的。我们 debug 到后面你会发现,configurations 的值变小了。</li></ul>
因为,这一步经历了一遍筛选,@ConditionalOnXXX 中的所有条件都满足,该类才会生效。
Spring Boot 提供的条件注解有如下:<br><br><ul><li>@ConditionalOnBean:当容器里有指定 Bean 的条件下</li><li>@ConditionalOnMissingBean:当容器里没有指定 Bean 的情况下</li><li>@ConditionalOnSingleCandidate:当指定 Bean 在容器中只有一个,或者虽然有多个但是指定首选 Bean</li><li>@ConditionalOnClass:当类路径下有指定类的条件下</li><li>@ConditionalOnMissingClass:当类路径下没有指定类的条件下</li><li>@ConditionalOnProperty:指定的属性是否有指定的值</li><li>@ConditionalOnResource:类路径是否有指定的值</li><li>@ConditionalOnExpression:基于 SpEL 表达式作为判断条件</li><li>@ConditionalOnJava:基于 Java 版本作为判断条件</li><li>@ConditionalOnJndi:在 JNDI 存在的条件下差在指定的位置</li><li>@ConditionalOnNotWebApplication:当前项目不是 Web 项目的条件下</li><li>@ConditionalOnWebApplication:当前项目是 Web 项 目的条件下</li></ul>
如何实现一个 Starter<br>
光说不练假把式,现在就来撸一个 starter,实现自定义线程池
第一步,创建 threadpool-spring-boot-starter工程<br>
第二步,引入 Spring Boot 相关依赖<br>
第三步,创建 ThreadPoolAutoConfiguration<br>
第四步,在 threadpool-spring-boot-starter工程的 resources 包下创建 META-INF/spring.factories 文件
最后新建工程引入 threadpool-spring-boot-starter
测试通过!!!
0 条评论
下一页