blog/source/_posts/spring-boot-starter.md

173 lines
9.0 KiB
Markdown
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

---
title: 编写Spring Boot Starter
tags:
- JVM
- Java
- Spring
- Spring Boot
categories:
- - JVM
- Java
- - JVM
- Spring
date: 2021-04-16 17:43:51
---
在使用Spring Boot的过程中许多依赖库都是以Spring Boot Starter的形式被引入的。实际上每一个Starter都是一个Spring Boot插件用来自动装配依赖库工作所需。通过Starter的工作可以简化Spring框架之前复杂的配置无论是基于XML的配置亦或是基于注解的配置。
<!-- more -->
通过使用Starter一个依赖库的依赖和配置都被集中在了一起。这样一来主项目就无需再逐个书写其所需要的依赖了而且也不必再逐一对依赖库进行手工装配了。Spring Boot会自动扫描Starter中的信息并加载、启动相应的配置。例如如果需要在项目中使用JDBC那么只需要在项目构建配置中引入`spring-boot-starter-jdbc`即可无需再去逐一寻找要使用JDBC所需要的各个依赖库。
## 一个Starter需要实现的功能
Spring官方所提供的Starter虽然已经能够满足日常项目开发的需要但是对于一些比较偏门的技术栈还是需要自行编写Starter的。毕竟每个项目中都书写一大片重复的依赖项任谁都不会爽。
那么编写一个Starter都需要实现哪些功能呢?
- 整合所需要的依赖项集成到Starter的构建配置中。
- 提供一套用于初始化依赖项的默认配置,这些默认配置可以在使用时被选择性覆盖。
- 使用自动化配置类对模块中的Bean进行自动装配并注入到Spring容器中。
## Starter的命名规则
Spring官方定义的Starter通常会遵循`spring-boot-starter-{dependency}`的命名格式,例如`spring-boot-starter-redis`。但是Spring官方同样也建议自定义的非官方Starter应该遵循`{dependency}-spring-boot-starter`的格式这样就会与官方提供的Starter有所区别但又不至于难以记忆。
## 开发一个Starter
Spring Boot Starter实际上与一个普通的Spring Boot应用没有太多的区别除了正常的编写构建配置和项目默认配置以外主要的工作都集中在了编写使用`@Configuration`和`@Bean`修饰的自动装配类,以及使用`spring.factories`文件定义的自动装配类。
在这里就不得不提一下Spring Boot的自动装配。Spring Boot在Spring Framework的基础上定义了一套接口规范这就是Spring Boot应用在启动的时候会首先扫描所有引用Jar包中的`META-INF/spring.factories`文件将这个文件中描述的类型信息加载到Spring容器中并同时完成其中定义的初始化工作。所以只要按照Spring Boot提供的接口规范编写就可以完成自定义的Starter。
下面以jetcd的Starter为例给出一个能够自动装配Etcd客户端的示例。首先Spring Boot应用在启动的时候会首先扫描`META-INF/spring.factories`文件来获取装配类。
```ini META-INF/spring.factories
org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
xyz.archgrid.jetcd.starter.EtcdClientConfiguration
```
如果Starter中有多个自动装配类可以使用逗号分隔列举在这里。
之后为了收集Etcd服务的连接信息还需要定义一个配置项收集类。
```java EtcdConfigurationProperties.java
@ConfigurationProperties(prefix = "etcd")
@Data
public class EtcdConfigurationProperties {
private Collection<String> endpoints = Collections.emptyList();
private String username = "";
private String password = "";
private String namespace = "";
private Long retryDelay = 20L;
private Integer connectTimeout = 60;
}
```
这个配置项收集类没有什么特殊的与平时在Spring Boot中使用的配置项收集类是一样的。在定义好配置项之后就是定义最关键的自动装配类了。
```java EtcdClientConfiguration.java
@Configuration
@ConditionalOnClass(Client.class)
@EnableConfigurationProperties(EtcdConfigurationProperties.class)
public class EtcdClientConfiguration {
private EtcdConfigurationProperties properties;
public EtcdClientConfiguration(EtcdConfigurationProperties properties) {
this.properties = properties;
}
@Bean
@Scope("singleton")
@ConditionalOnMissingBean(Client.class)
public Client buildEtcdClient() {
ClientBuilder builder = Client.builder();
if (properties.getEndpoints().isEmpty()) {
properties.getEndpoints().add("http://127.0.0.1:2379");
}
builder.endpoints(properties.getEndpoints().toArray(new String[0]));
if (!properties.getUsername().isBlank()) {
builder.user(ByteSequence.from(properties.getUsername(), StandardCharsets.UTF_8));
}
if (!properties.getPassword().isBlank()) {
builder.password(ByteSequence.from(properties.getPassword(), StandardCharsets.UTF_8));
}
if (!properties.getNamespace().isBlank()) {
builder.namespace(ByteSequence.from(properties.getNamespace(), StandardCharsets.UTF_8));
}
builder.retryChronoUnit(ChronoUnit.SECONDS);
builder.retryDelay(properties.getRetryDelay());
builder.connectTimeoutMs(properties.getConnectTimeout() * 1000);
return builder.build();
}
@Bean
@Scope("prototype")
@ConditionalOnMissingBean(KV.class)
public KV buildEtcdKVClient(Client client) {
return client.getKVClient();
}
}
```
这个自动装配类其实也就是一个普通的Spring Boot的配置类只是在它上面增加了一些条件注解。这些条件注解可以根据Spring对象池内的对象来决定是否要使用制动装配类中定义的工厂方法组建新的对象实例。除了示例中使用的`@ConditionalOnClass`和`@ConditionalOnMissingBean`以外,常用的条件注解主要有以下这些。
- `@ConditionalOnMissingBean`当指定的条件的Bean不存在的时候启动装配。通常用于补充应用中需要的Bean避免重复装配。
- `@ConditionalOnBean`当指定的条件的Bean存在的时候启动装配。通常用于根据Bean之间的依赖来避免装配失败。
- `@ConditionalOnMissingClass`当指定的条件的类不存在的时候启动装配。通常用于在应用中提供替代Bean。
- `@ConditionalOnClass`,当指定的条件的类存在的时候启动装配。通常用于在应用中根据类之间的依赖避免装配失败。
- `@ConditionalOnProperty`,根据指定配置文件中的配置项来确定是否启动装配。
- `@ConditionalOnResource`,根据指定的资源是否存在确定是否启动装配。
- `@ConditionalOnExpression`根据SpEL表达式的执行结果确定是否启动装配。
## 自动装配机制简述
Spring Boot应用所使用的`@SpringBootApplication`注解,实际上可以被看作是`@Configuration`、`@EnableAutoConfiguration`和`@ComponentScan`三个注解的组合。这三个注解分别用于注册额外的Bean或者导入其他的配置类、启动Spring的自动装配机制和扫描被`@component`注解的Bean。其中`@EnableAutoConfiguration`注解可以配合`AutoConfigurationImportSelector`类来实现自动装配功能。
## 一个小技巧
在IntelliJ IDEA中对Spring Boot应用的`application.properties`或者`application.yml`文件进行编辑的时候往往可以看到针对目前应用所使用的Starter提供的配置项提示。实际上这是Starter中的文件`META-INF/spring-configuration-metadata.json`发挥的作用这个文件的内容描述了Starter中所需要的全部配置项的信息。
这个元信息描述文件主要需要编写的内容是各个配置项的名称、类型以及描述对于配置项的类型可以直接使Java的类型全称来描述。对于前面这个用于Etcd的Starter来说可以编写一个下面这样的描述文件。
```json META-INF/spring-configuration-metadata.json
{
"properties": [
{
"name": "etcd.endpoints",
"type": "java.util.Collection<java.lang.String>",
"description": "Etcd主机列表"
},
{
"name": "etcd.username",
"type": "java.lang.String",
"description": "Etcd主机认证用户名",
"defaultValue": ""
},
{
"name": "etcd.password",
"type": "java.lang.String",
"description": "Etcd主机认证密码",
"defaultValue": ""
},
{
"name": "etcd.namespace",
"type": "java.lang.String",
"description": "所要操作的默认命名空间",
"defaultValue": ""
},
{
"name": "etcd.retryDelay",
"type": "java.lang.Long",
"description": "连接失败后重试的时间间隔,单位秒",
"defaultValue": 20
},
{
"name": "etcd.connectTimeout",
"type": "java.lang.Integer",
"description": "连接超时时间,单位秒",
"defaultValue": 60
}
]
}
```