【Java教程】Spring Boot 中starter原理详解

所需工具:

Java

聪明的大脑

勤劳的双手

 

注意:本站只提供教程,不提供任何成品+工具+软件链接,仅限用于学习和研究,禁止商业用途,未经允许禁止转载/分享等

 

教程如下

前言:

今天介绍springboot ,也是写下springboot的插件机制,starter的原理,其实这个网上已经很多了,也是看了不少别人的文章,今天主要还是带着问题去记录下。

1、springboot 的starter 的启动原理是什么

原理

这个问题是很简单的,只要了解springboot的同学应该都知道,也是必须了解的。

① Spring Boot 在启动时会去classpath中中寻找 resources/META-INF/spring.factories 文件
② 根据 spring.factories 配置加载 AutoConfigure 类
③ 根据 @Conditional 注解的条件,进行自动配置并将 Bean 注入 Spring Context

来个例子

我们看个例子,下面是我工作中的一个starter,删除了一些公司的代码

 	org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
 	com.xx.common.ratelimiter.config.reactive.WebsocketRateLimiterAutoConfiguration,\
 	com.xx.common.ratelimiter.config.ProductIdLimiterAutoConfiguration

解释下:

EnableAutoConfiguration 这个是固定的,在springboot启动的时候,会去实例化这个key之后的每一个类
\ 是连接符,也就是表示这是一行数据,只不过可以分行表示
WebsocketRateLimiterAutoConfiguration 后面这两行是业务自定义的config类,多个的话都好分割

 	@Configuration
 	@ConditionalOnProperty(
 	         prefix = "gateway.ratelimiter",
 	         name = {"enabled"},
 	         havingValue = "true"
 	)
 	@EnableConfigurationProperties(RateLimiterProperties.class)
 	@AutoConfigureAfter({RedisAutoConfiguration.class})
 	public class ProductIdLimiterAutoConfiguration {
 	...
 	}

上面这个是配置类,可以看到这个最主要的是一些注解

@Configuration 标识这是一个配置类
@ConditionalOnProperty 标识这个类的实例化需要在配置文件中存在 gateway.ratelimiter.enable 这个key,并且值为true
@EnableConfigurationProperties 将配置中的类,读取到RateLimiterProperties.class 这个对象中
@AutoConfigureAfter 这个顾名思义,需要在redis 的实例化之后,因为这个config依赖redis 。

小结

springboot starter的原理就是springboot项目在启动的时候扫描jar,然后读取spring.factories 中EnableAutoConfiguration key 指向的config类

然后通过一系列的条件配置判断,启动当前stater

2、springboot 是如何找到配置类的

第一个问题解决了主要流程的问题,但是springboo是怎么找到spring.fatories的呐、

这个主要是涉及jar包中资源的读取,让我们看下这个流程。

主要的路径是下面这三个注解:

 	SpringBootApplication ->
 	     EnableAutoConfiguration ->
 	         AutoConfigurationImportSelector org.springframework.boot.autoconfigure.AutoConfigurationImportSelector#selectImports
 	         org.springframework.boot.autoconfigure.AutoConfigurationImportSelector#getAutoConfigurationEntry
 	         org.springframework.boot.autoconfigure.AutoConfigurationImportSelector#getCandidateConfigurations
 	         org.springframework.core.io.support.SpringFactoriesLoader#loadFactoryNames
 	         org.springframework.core.io.support.SpringFactoriesLoader#loadSpringFactories Enumeration<URL> urls = classLoader.getResources(FACTORIES_RESOURCE_LOCATION);
 	         public static final String FACTORIES_RESOURCE_LOCATION = "META-INF/spring.factories";

通过上面的路径,我们找到了最终是通过classloader 加载jar中”META-INF/spring.factories”;当然这段是代码我们也可以用来加载其他的文件哦,也给我们借鉴,在下次需要实现类似的功能,可以直接抄。

3、springboot starter 的bean 是怎么加载到容器的

这个问题已经很简单了,因为第一个问题已经基本上可以看到原因了,

在springboot 加载到config的时候,可以在config中通过@bean进行容器注册,这个bean就会加载到容器

这里主要要说几个特殊的注解,@ConditionalOnXXX

@ConditionalOnBean:当容器中有指定Bean的条件下进行实例化。
@ConditionalOnMissingBean:当容器里没有指定Bean的条件下进行实例化。
@ConditionalOnClass:当classpath类路径下有指定类的条件下进行实例化。
@ConditionalOnMissingClass:当类路径下没有指定类的条件下进行实例化。
@ConditionalOnWebApplication:当项目是一个Web项目时进行实例化。
@ConditionalOnNotWebApplication:当项目不是一个Web项目时进行实例化。
@ConditionalOnProperty:当指定的属性有指定的值时进行实例化。
@ConditionalOnExpression:基于SpEL表达式的条件判断。
@ConditionalOnjava:当JVM版本为指定的版本范围时触发实例化。
@ConditionalOnResource:当类路径下有指定的资源时触发实例化。
@ConditionalOnJndi:在JNDI存在的条件下触发实例化。
@ConditionalOnSingleCandidate:当指定的Bean在容器中只有一个,或者有多个但是指定了首选的Bean时触发实例化。

如果想要定制自己的condition,可以实现Condition接口,定制Condition条件

4、总结

springboot starter 是springboot 的核心,提供了autoconfig,在开发中非常方便,也是必须需要了解的。

标签

发表评论