当前位置: 首页 > news >正文

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

目录

  • 一、SpringBoot Starter 简介
    • 1、什么是 SpringBoot Starter
    • 2、什么场景下需要自定义 Starter
  • 二、创建自定义的 SpringBoot Starter
    • 1、Starter 命令规范
    • 2、Starter 创建步骤
    • 3、测试验证

一、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

相关文章:

  • 阶段一:Java基础入门
  • 西门子——好用的通讯仿真通讯工具NetToPLCsim
  • docker对网络和程序速度的影响
  • 下沉一线农技志愿服务 国稻种芯-芜湖:湾沚红杨护秋粮生产
  • java计算机毕业设计同学录管理系统源码+系统+数据库+lw文档+mybatis+运行部署
  • 枯竭的水库求生的稻田 国稻种芯·九江:位于抗旱一线的都昌
  • 网课搜题公众号题库接口系统
  • 分享网课查题公众号接口 及如何使用
  • 网课搜题公众号制作方法
  • Ubuntu 安装 HDF5 C++库
  • 【Linux系统管理】10 Shell 编程进阶篇
  • Jackson ImmunoResearch 直接和间接蛋白质印迹方案
  • bazel构建项目案例(第三方库,编译成库,运行案例)
  • Jackson ImmunoResearch 用于蛋白质印迹的偶联物方案
  • 如何扫描到最新可用的http代理ip?
  • Javascript设计模式学习之Observer(观察者)模式
  • mongodb--安装和初步使用教程
  • vue-cli在webpack的配置文件探究
  • Webpack入门之遇到的那些坑,系列示例Demo
  • WinRAR存在严重的安全漏洞影响5亿用户
  • 分享几个不错的工具
  • 基于MaxCompute打造轻盈的人人车移动端数据平台
  • 聊聊redis的数据结构的应用
  • 原生 js 实现移动端 Touch 滑动反弹
  • 字符串匹配基础上
  • MyCAT水平分库
  • ​ 轻量应用服务器:亚马逊云科技打造全球领先的云计算解决方案
  • ​力扣解法汇总946-验证栈序列
  • # Apache SeaTunnel 究竟是什么?
  • #、%和$符号在OGNL表达式中经常出现
  • #单片机(TB6600驱动42步进电机)
  • #我与Java虚拟机的故事#连载06:收获颇多的经典之作
  • (1)(1.8) MSP(MultiWii 串行协议)(4.1 版)
  • (Mirage系列之二)VMware Horizon Mirage的经典用户用例及真实案例分析
  • (二十三)Flask之高频面试点
  • (牛客腾讯思维编程题)编码编码分组打印下标(java 版本+ C版本)
  • (算法)求1到1亿间的质数或素数
  • (未解决)jmeter报错之“请在微信客户端打开链接”
  • .mat 文件的加载与创建 矩阵变图像? ∈ Matlab 使用笔记
  • .Net 6.0 处理跨域的方式
  • .NET LINQ 通常分 Syntax Query 和Syntax Method
  • .net 提取注释生成API文档 帮助文档
  • .net和php怎么连接,php和apache之间如何连接
  • .pop ----remove 删除
  • .pyc文件是什么?
  • @TableId注解详细介绍 mybaits 实体类主键注解
  • [ 手记 ] 关于tomcat开机启动设置问题
  • [Angular 基础] - 数据绑定(databinding)
  • [AR]Vumark(下一代条形码)
  • [C语言]——内存函数
  • [Deep Learning] 神经网络基础
  • [DevOps云实践] 彻底删除AWS云资源
  • [Excel] vlookup函数
  • [HDU5685]Problem A
  • [Hive] CTE 通用表达式 WITH关键字