【SpringBoot】之创建自定义 SpringBoot-Starter

目录


一、SpringBoot Starter 简介


1、什么是 SpringBoot Starter

SpringBoot 的出现让我们能够抛弃以前复杂繁杂的配置,转而将组件各种配置统一集成进 starter 中,在使用的使用只需要在 maven 中引入 starter 依赖,SpringBoot 就能自动扫描到要加载的信息并启动相应的默认配置。

Starter 让我们摆脱了各种依赖库的处理以及需要配置各种信息的困扰。因为 SpringBoot 会自动通过 classpath 路径下的类发现需要的 Bean,对其进行自动装配并注册进 IOC 容器。

SpringBoot 提供了许多我们日常开发中在各种场景所需的 spring-boot-starter 依赖模块。所有这些依赖模块都会遵循着约定成俗的默认配置,并允许我们调整这些配置,即遵循约定大于配置的理念。

想了解更多关于 SpringBoot 的知识以及 SpringBoot 是如何进行自动装配组件的,可以参考我的另一篇博客:SpringBoot】之自动配置原理分析(源码级别)

2、什么场景下需要自定义 Starter

在日常的开发工作中,经常会有一些独立于业务之外的配置模块,如果我们将其放到一个特定的包下,然后另一个工程需要复用这块功能的时候,就需要将代码硬拷贝到另一个工程,重新集成一遍,这样做非常的麻烦。

但如果我们将这些可独立于业务代码之外的功配置模块封装成独立的 Starter,复用的时候只需要将其在 pom 中引用依赖即可,因为 SpringBoot 的自动装配会为我们完成模块的装配,极大地方便了我们开发。

常见场景如下:

  • 构建通用模块:比如短信、邮件发送模块
  • 基于 AOP 技术实现日志切面
  • 处理分布式雪花 ID:将 Long 型转为 String 型、使用 jackson2/fastjson 解决精度问题

二、创建自定义的 SpringBoot Starter


1、Starter 命令规范

每个 Starter 都会遵循标准的命名规范,其中分为官方 Starter 的命名方法和自定义 Starter 的命名方式:

SpringBoot 官方命名方式:

  • 格式:spring-boot-starter-{模块名}
  • 举例:spring-boot-starter-web

自定义命名方式:

  • 格式:{模块名}-spring-boot-starter
  • 举例:mystarter-spring-boot-starter

2、Starter 创建步骤

我们先回顾一下 SpringBoot 的自动装配原理:

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

这个原理相当于给了我们一个借鉴,只要我们遵循了这个规范我们就可以很简单的进行创建自定义的 Starter

步骤如下:

1)创建 Maven 项目

创建一个空的 Maven 项目:

Starter 不需要 main 启动类,一般创建空的 Maven 项目最合适,如果有生成启动类,直接删除掉。

2)添加依赖

下面市场完整的依赖:

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.2.1.RELEASE</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>

    <groupId>com.wty</groupId>
    <artifactId>email-spring-boot-starter</artifactId>
    <version>1.0.0-SNAPSHOT</version>

    <properties>
        <maven.compiler.source>8</maven.compiler.source>
        <maven.compiler.target>8</maven.compiler.target>
    </properties>

    <dependencies>
        <!--包含自动配置的代码-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-autoconfigure</artifactId>
        </dependency>
        <!--非必须:编写配置文件时会有提示-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-configuration-processor</artifactId>
            <optional>true</optional>
        </dependency>
        
        <!--下面就是按照自己项目所需,引入相关依赖-->
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <optional>true</optional>
        </dependency>
    </dependencies>

</project>

这里需要注意两点:

  • spring-boot-configuration-processor 并不是必须的,只是引入之后,在 yml 配置文件中编写配置时会有提示;而设置 <optional> 标签的值为 true 表示,两个项目直接依赖不传递,例如:项目A依赖 a.jar,项目B依赖项目A,则项目B不依赖 a.jar;
  • 由于 starter 是没有 main 方法入口的,所以需要去除 pom 文件中 maven 打包插件 spring-boot-maven-plugin。

2)编写相关属性配置类

@Data
@ConfigurationProperties(prefix = "email.config")
public class EmailProperties {

    private String username = "";

    private String password = "";

    private String host = "smtp.163.com";

    private Integer port = 25;

}

3)编写 Starter 的功能类

@Data
public class EmailService {

    private EmailProperties emailProperties;

    /**
     * 模拟邮件发送功能
     */
    public void sendEmail() {
        System.out.println("Sending email... \n" +
                "username: " + emailProperties.getUsername() + "\n" +
                "password: " + emailProperties.getPassword() + "\n" +
                "host: " + emailProperties.getHost() + "\n" +
                "port: " + emailProperties.getPort());
    }

}

4)编写自动配置类

@Configuration(proxyBeanMethods = false)
@ConditionalOnClass({EmailService.class})
@EnableConfigurationProperties({EmailProperties.class})
public class EmailAutoConfiguration {

    @Autowired
    private EmailProperties emailProperties;

    @Bean
    @ConditionalOnMissingBean
    public EmailService emailService() {
        EmailService emailService = new EmailService();
        emailService.setEmailProperties(emailProperties);
        return emailService;
    }

}

解析:

  • @Configuration 注解修饰表示该类为配置类,会注入到容器中,proxyBeanMethods = false 表示使用 Lite 轻量级模式
  • @ConditionalOnClass 注解表示只有存在 EmailService.class 类时才生效自动配置类;
  • @EnableConfigurationProperties 注解使我们定义的 EmailProperties.class 属性配置类生效。

扩展知识:Full 全模式和 Lite 轻量级模式

@Configuration 注解的参数 proxyBeanMethods 表示设置 Bean 的代理模式:

  • Full 全模式(proxyBeanMethods = true,默认值):同一配置类下,当直接调用 @Bean 修饰的方法注入的对象,则调用该方法会被代理,从 ioc 容器中取 bean 实列,所以实列是一样的。即单实例对象,在该模式下 SpringBoot 每次启动都会判断检查容器中是否存在该组件;
  • Lite 轻量级模式(proxyBeanMethods = false):同一配置类下,当直接调用 @Bean 修饰的方法注入的对象,则调用该方法不会被代理,相当于直接调用一个普通方法,会有构造方法,但是没有 bean 的生命周期,返回的是不同的实例。

注意: proxyBeanMethods 只是为了让使用 @Bean 注解的方法被代理,而不是对 @Bean 设置其为单例还是多例。

什么时候用Full全模式,什么时候用Lite轻量级模式:

  • 当在你的同一个 Configuration 配置类中,注入到容器中的 bean 实例之间有依赖关系时,建议使用 Full 全模式;
  • 当在你的同一个 Configuration 配置类中,注入到容器中的 bean 实例之间没有依赖关系时,建议使用 Lite 轻量级模式,以提高 springboot 的启动速度和性能。

5)编写 spring.factories 文件

在 src/main/resources 下添加 META-INF/spring.factories 文件

org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
  com.wty.starter.config.EmailAutoConfiguration

把自动配置类 EmailAutoConfiguration 配置到org.springframework.boot.autoconfigure.EnableAutoConfiguration 的 key 下,SpringBoot 会自动加载该文件并根据条件装配。

6)打包

在 Maven 插件的 Lifecycle 下点击 install,对项目进行打包生成 starter 包:email-spring-boot-starter

3、测试验证

在另一个项目中引入我们自定义的 starter 依赖:

<dependency>
    <groupId>com.wty</groupId>
    <artifactId>email-spring-boot-starter</artifactId>
    <version>1.0.0-SNAPSHOT</version>
</dependency>

在 yml 配置相应的属性(编写的过程中会弹出属性配置提示):

email:
  config:
    username: xxx@163.com
    password: 123456
    host: smtp.163.com
    port: 25

编写测试接口:

@RestController
@RequestMapping("/starter")
public class StarterController {

    @Autowired
    private EmailService emailService;

    @GetMapping("/email")
    public void testEmail() {
        emailService.sendEmail();
    }

}

得到结果:

Sending email... 
username: xxx@163.com
password: 123456
host: smtp.163.com
port: 25
  • 6
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
自定义Spring Boot Starter是一种用于简化Spring Boot应用程序配置和依赖管理的机制。它由几个组件组成,包括starter包、autoconfiguration包和配置文件。 首先,你需要创建一个Maven工程,并创建三个模块。其中,starter包负责引入依赖,autoconfiguration包负责实现装配。在autoconfiguration包中,你需要定义对应的properties类、configuration类和核心业务类。此外,你还需要在autoconfiguration包的/resources/META-INF/目录下添加spring.factories文件,并配置org.springframework.boot.autoconfigure.EnableAutoConfiguration=com.qiejk.demo.springboot.autoconfiguration.DemoAutoConfiguration。这一步是Spring Boot装配的核心,只有添加了这个内容,Spring Boot才会进行自动装配。\[1\] 在父模块的pom文件中,你需要引入Spring Boot的依赖,例如spring-boot-starter-web,以及指定Spring Boot的版本。这样,你的自定义starter模块就可以基于Spring Boot进行开发和使用。\[2\] 最后,你需要创建一个配置文件,用于告诉Spring Boot在启动时扫描该配置文件,并将其中的配置类加载到Spring容器中。这个配置文件的作用是指定自动配置类的位置。\[3\] 通过以上步骤,你就可以创建自定义Spring Boot Starter,并在应用程序中使用它来简化配置和依赖管理。 #### 引用[.reference_title] - *1* [如何自定义springboot-starter](https://blog.csdn.net/sinat_29434159/article/details/123995794)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* *3* [SpringBoot自定义Starter篇](https://blog.csdn.net/m0_46571920/article/details/122910726)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值