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

[[380163]]

starter背景

Spring Boot目前已经变成了后端开发这必备技能之一,其中一个主要原因是Spring Boot中有个非常重要的机制(starter机制)。

starter能够抛弃以前繁杂的配置,将其统一集成进starter,使用的时候只需要在maven中引入对应的starter依赖即可,Spring Boot就能自动扫描到要加载的信息并启动相应的默认配置。

starter让我们摆脱了各种依赖库的处理,以及各种配置信息的烦恼。SpringBoot会自动通过classpath路径下的类发现需要的Bean,并注册进IOC容器。Spring Boot提供了针对日常企业应用研发各种场景的spring-boot-starter依赖模块。所有这些依赖模块都遵循着约定成俗的默认配置,并允许我们调整这些配置,即遵循“约定大于配置”的理念。

我们经常会看到或者使用到各种xxx-starter。比如下面几种:

Spring Boot starter原理

从总体上来看,无非就是将Jar包作为项目的依赖引入工程。而现在之所以增加了难度,是因为我们引入的是Spring Boot Starter,所以我们需要去了解Spring Boot对Spring Boot Starter的Jar包是如何加载的?下面我简单说一下。

SpringBoot 在启动时会去依赖的 starter 包中寻找 /META-INF/spring.factories 文件,然后根据文件中配置的 Jar 包去扫描项目所依赖的 Jar 包,这类似于 Java 的 SPI 机制。

细节上可以使用@Conditional 系列注解实现更加精确的配置加载Bean的条件。

JavaSPI 实际上是“基于接口的编程+策略模式+配置文件”组合实现的动态加载机制。

自定义starter的条件

如果想自定义Starter,首选需要实现自动化配置,而要实现自动化配置需要满足以下两个条件:

  1. 能够自动配置项目所需要的配置信息,也就是自动加载依赖环境;
  2. 能够根据项目提供的信息自动生成Bean,并且注册到Bean管理容器中;

实现自定义starter

  1. <dependencies> 
  2.  <dependency> 
  3.     <groupId>org.springframework.boot</groupId> 
  4.     <artifactId>spring-boot-autoconfigure</artifactId> 
  5.     <version>2.0.0.RELEASE</version> 
  6.  </dependency> 
  7.  <dependency> 
  8.     <groupId>org.springframework.boot</groupId> 
  9.     <artifactId>spring-boot-configuration-processor</artifactId> 
  10.     <version>2.0.0.RELEASE</version> 
  11.     <optional>true</optional> 
  12.   </dependency> 
  13. </dependencies> 

根据需要自定义Starter的实现过程大致如下(以我定义的Starter为例):

定义XxxProperties类,属性配置类,完成属性配置相关的操作,比如设置属性前缀,用于在application.properties中配置。

TianProperties代码:

  1. import org.springframework.boot.context.properties.ConfigurationProperties; 
  2. @ConfigurationProperties(prefix = "spring.tian"
  3. public class TianProperties { 
  4.     private String name
  5.     private int age; 
  6.     private String sex = "M"
  7.     //省略 get set 方法 

创建XxxService类,完成相关的操作逻辑 。

TianService代码:

  1. public class TianService { 
  2.  
  3.     private TianProperties properties; 
  4.  
  5.     public TianService() { 
  6.     } 
  7.  
  8.     public TianService(TianProperties userProperties) { 
  9.         this.properties = userProperties; 
  10.     } 
  11.     public void sayHello(){ 
  12.         System.out.println("hi, 我叫: " + properties.getName() + 
  13.         ", 今年" + properties.getAge() + "岁" 
  14.          + ", 性别: " + properties.getSex()); 
  15.     } 

定义XxxConfigurationProperties类,自动配置类,用于完成Bean创建等工作。

TianServiceAutoConfiguration代码:

  1. @Configuration 
  2. @EnableConfigurationProperties(TianProperties.class) 
  3. @ConditionalOnClass(TianService.class) 
  4. @ConditionalOnProperty(prefix = "spring.tian", value = "enabled", matchIfMissing = true
  5. public class TianServiceAutoConfiguration { 
  6.  
  7.     @Autowired 
  8.     private TianProperties properties; 
  9.  
  10.     @Bean 
  11.     @ConditionalOnMissingBean(TianService.class) 
  12.     public TianService tianService() { 
  13.         return new TianService(properties); 
  14.     } 

在resources下创建目录META-INF,在 META-INF 目录下创建 spring.factories,在SpringBoot启动时会根据此文件来加载项目的自动化配置类。

「spring.factories中配置」

  1. org.springframework.boot.autoconfigure.EnableAutoConfiguration=com.tian.TianServiceAutoConfiguration 

把上面这个starter工程打成jar包:

使用自定义starter

创建一个Spring Boot项目test,项目整体如下图:

在项目中把自定义starter添加pom依赖

  1. <dependency> 
  2.     <groupId>com.tian</groupId> 
  3.     <artifactId>spring-boot-tian-starter</artifactId> 
  4.     <version>1.0-SNAPSHOT</version> 
  5. </dependency> 

TestApplication启动类

  1. @SpringBootApplication 
  2. @EnableEurekaServer 
  3. public class TestApplication { 
  4.     public static void main(String[] args) { 
  5.         SpringApplication.run(TestApplication.class, args); 
  6.     } 

application.properties中配置

  1. spring.tian.name=tian 
  2. spring.tian.age=22 
  3. spring.tian.sex=M 

写一个TestController.java类

  1. RestController 
  2. @RequestMapping("/my"
  3. public class TestController { 
  4.  
  5.     @Resource 
  6.     private TianService tianService; 
  7.  
  8.     @PostMapping("/starter"
  9.     public Object starter() { 
  10.         tianService.sayHello(); 
  11.         return "ok"
  12.     } 

把我们自定义的starter打成的jar依赖进来后,

可以看到其中多了一个json的文件。

最后启动项目,输入

http://localhost:9091/my/starter

controller成功返回ok,再看后台打印

  1. hi, 我叫: tian, 今年22岁, 性别: M 

这就成功的现实了自定义的starter。

关键词:开箱即用、减少大量的配置项、约定大于配置。

总结

  1. Spring Boot在启动时扫描项目所依赖的JAR包,寻找包含spring.factories文件的JAR包,
  2. 然后读取spring.factories文件获取配置的自动配置类AutoConfiguration`,
  3. 然后将自动配置类下满足条件(@ConditionalOnXxx)的@Bean放入到Spring容器中(Spring Context)
  4. 这样使用者就可以直接用来注入,因为该类已经在容器中了。

本文转载自微信公众号「Java后端技术全栈」,可以通过以下二维码关注。转载本文请联系Java后端技术全栈公众号。

 

文章来源网络,作者:运维,如若转载,请注明出处:https://shuyeidc.com/wp/152984.html<

(0)
运维的头像运维
上一篇2025-03-14 06:24
下一篇 2025-03-14 06:26

相关推荐

  • 个人主题怎么制作?

    制作个人主题是一个将个人风格、兴趣或专业领域转化为视觉化或结构化内容的过程,无论是用于个人博客、作品集、社交媒体账号还是品牌形象,核心都是围绕“个人特色”展开,以下从定位、内容规划、视觉设计、技术实现四个维度,详细拆解制作个人主题的完整流程,明确主题定位:找到个人特色的核心主题定位是所有工作的起点,需要先回答……

    2025-11-20
    0
  • 社群营销管理关键是什么?

    社群营销的核心在于通过建立有温度、有价值、有归属感的社群,实现用户留存、转化和品牌传播,其管理需贯穿“目标定位-内容运营-用户互动-数据驱动-风险控制”全流程,以下从五个维度展开详细说明:明确社群定位与目标社群管理的首要任务是精准定位,需明确社群的核心价值(如行业交流、产品使用指导、兴趣分享等)、目标用户画像……

    2025-11-20
    0
  • 香港公司网站备案需要什么材料?

    香港公司进行网站备案是一个涉及多部门协调、流程相对严谨的过程,尤其需兼顾中国内地与香港两地的监管要求,由于香港公司注册地与中国内地不同,其网站若主要服务内地用户或使用内地服务器,需根据服务器位置、网站内容性质等,选择对应的备案路径(如工信部ICP备案或公安备案),以下从备案主体资格、流程步骤、材料准备、注意事项……

    2025-11-20
    0
  • 如何企业上云推广

    企业上云已成为数字化转型的核心战略,但推广过程中需结合行业特性、企业痛点与市场需求,构建系统性、多维度的推广体系,以下从市场定位、策略设计、执行落地及效果优化四个维度,详细拆解企业上云推广的实践路径,精准定位:明确目标企业与核心价值企业上云并非“一刀切”的方案,需先锁定目标客户群体,提炼差异化价值主张,客户分层……

    2025-11-20
    0
  • PS设计搜索框的实用技巧有哪些?

    在PS中设计一个美观且功能性的搜索框需要结合创意构思、视觉设计和用户体验考量,以下从设计思路、制作步骤、细节优化及交互预览等方面详细说明,帮助打造符合需求的搜索框,设计前的规划明确使用场景:根据网站或APP的整体风格确定搜索框的调性,例如极简风适合细线条和纯色,科技感适合渐变和发光效果,电商类则可能需要突出搜索……

    2025-11-20
    0

发表回复

您的邮箱地址不会被公开。必填项已用 * 标注