创建您自己的自动配置
如果您在一家开发共享库的公司工作,或者您从事开源或商业库的开发,您可能希望开发您自己的自动配置。自动配置类可以打包在外部 jar 中,并且仍然会被 Spring Boot 拾取。
自动配置可以与“启动器”相关联,启动器提供自动配置代码以及您通常与其一起使用的典型库。我们首先介绍构建您自己的自动配置所需的知识,然后继续介绍 创建自定义启动器所需的典型步骤。
了解自动配置的 Bean
实现自动配置的类使用@AutoConfiguration
注解进行标注。该注解本身使用@Configuration
进行元注解,使自动配置成为标准的@Configuration
类。此外,使用额外的@Conditional
注解来约束自动配置何时应用。通常,自动配置类使用@ConditionalOnClass
和@ConditionalOnMissingBean
注解。这确保了只有在找到相关类并且您没有声明自己的@Configuration
时,自动配置才会应用。
您可以浏览spring-boot-autoconfigure
的源代码,以查看Spring提供的@AutoConfiguration
类(请参阅META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports
文件)。
查找自动配置候选者
Spring Boot 检查您的已发布 jar 文件中是否存在META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports
文件。该文件应列出您的配置类,每行一个类名,如下例所示
com.mycorp.libx.autoconfigure.LibXAutoConfiguration
com.mycorp.libx.autoconfigure.LibXWebAutoConfiguration
您可以使用# 字符在 imports 文件中添加注释。 |
自动配置**必须**且**仅能**通过在 imports 文件中命名来加载。确保它们定义在特定的包空间中,并且永远不会成为组件扫描的目标。此外,自动配置类不应启用组件扫描以查找其他组件。应改为使用特定的@Import 注解。 |
如果您的配置需要按特定顺序应用,则可以在@AutoConfiguration
注解上或专用的@AutoConfigureBefore
和@AutoConfigureAfter
注解上使用before
、beforeName
、after
和afterName
属性。例如,如果您提供特定于 Web 的配置,则可能需要在WebMvcAutoConfiguration
之后应用您的类。
如果您想对某些不应该相互了解的自动配置进行排序,还可以使用@AutoConfigureOrder
。该注解与常规的@Order
注解具有相同的语义,但为自动配置类提供了专用的顺序。
与标准的@Configuration
类一样,自动配置类应用的顺序仅影响其 bean 定义的顺序。这些 bean 随后创建的顺序不受影响,并由每个 bean 的依赖项和任何@DependsOn
关系决定。
条件注解
您几乎总是希望在自动配置类上包含一个或多个@Conditional
注解。@ConditionalOnMissingBean
注解是一个常见的示例,用于允许开发人员在对默认值不满意时覆盖自动配置。
Spring Boot 包含许多@Conditional
注解,您可以通过注解@Configuration
类或单个@Bean
方法在自己的代码中重用这些注解。这些注解包括
类条件
@ConditionalOnClass
和@ConditionalOnMissingClass
注解允许根据特定类的存在或不存在来包含@Configuration
类。由于注解元数据是使用ASM解析的,因此您可以使用value
属性引用实际类,即使该类可能实际上并未出现在正在运行的应用程序类路径上。如果希望使用String
值指定类名,也可以使用name
属性。
此机制不以相同的方式应用于@Bean
方法,在@Bean
方法中,条件的目标通常是方法的返回类型:在方法上的条件应用之前,JVM 将加载该类并可能处理方法引用,如果该类不存在,则这些引用将失败。
要处理这种情况,可以使用单独的@Configuration
类来隔离条件,如下例所示
-
Java
-
Kotlin
import org.springframework.boot.autoconfigure.AutoConfiguration;
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@AutoConfiguration
// Some conditions ...
public class MyAutoConfiguration {
// Auto-configured beans ...
@Configuration(proxyBeanMethods = false)
@ConditionalOnClass(SomeService.class)
public static class SomeServiceConfiguration {
@Bean
@ConditionalOnMissingBean
public SomeService someService() {
return new SomeService();
}
}
}
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
// Some conditions ...
class MyAutoConfiguration {
// Auto-configured beans ...
@Configuration(proxyBeanMethods = false)
@ConditionalOnClass(SomeService::class)
class SomeServiceConfiguration {
@Bean
@ConditionalOnMissingBean
fun someService(): SomeService {
return SomeService()
}
}
}
如果将@ConditionalOnClass 或@ConditionalOnMissingClass 用作元注解的一部分来组合您自己的组合注解,则必须使用name ,因为在这种情况下,对类的引用不会被处理。 |
Bean 条件
@ConditionalOnBean
和@ConditionalOnMissingBean
注解允许根据特定 bean 的存在或不存在来包含 bean。您可以使用value
属性按类型指定 bean,或使用name
属性按名称指定 bean。search
属性允许您限制在搜索 bean 时应考虑的ApplicationContext
层次结构。
当放置在@Bean
方法上时,目标类型默认为方法的返回类型,如下例所示
-
Java
-
Kotlin
import org.springframework.boot.autoconfigure.AutoConfiguration;
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
import org.springframework.context.annotation.Bean;
@AutoConfiguration
public class MyAutoConfiguration {
@Bean
@ConditionalOnMissingBean
public SomeService someService() {
return new SomeService();
}
}
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean
import org.springframework.context.annotation.Bean
import org.springframework.context.annotation.Configuration
@Configuration(proxyBeanMethods = false)
class MyAutoConfiguration {
@Bean
@ConditionalOnMissingBean
fun someService(): SomeService {
return SomeService()
}
}
在前面的示例中,如果ApplicationContext
中尚未包含类型为SomeService
的 bean,则将创建someService
bean。
您需要非常小心 bean 定义添加的顺序,因为这些条件是根据到目前为止已处理的内容进行评估的。因此,我们建议仅在自动配置类上使用@ConditionalOnBean 和@ConditionalOnMissingBean 注解(因为可以保证这些注解在添加任何用户定义的 bean 定义之后加载)。 |
@ConditionalOnBean 和@ConditionalOnMissingBean 不会阻止创建@Configuration 类。在类级别使用这些条件与使用注解标记每个包含的@Bean 方法之间的唯一区别在于,如果条件不匹配,前者会阻止将@Configuration 类注册为 bean。 |
在声明@Bean 方法时,请在方法的返回类型中提供尽可能多的类型信息。例如,如果您的 bean 的具体类实现了接口,则 bean 方法的返回类型应为具体类,而不是接口。在@Bean 方法中提供尽可能多的类型信息在使用 bean 条件时尤其重要,因为它们的评估只能依赖于方法签名中可用的类型信息。 |
属性条件
@ConditionalOnProperty
注解允许根据 Spring Environment 属性来包含配置。使用prefix
和name
属性指定要检查的属性。默认情况下,任何存在的且不等于false
的属性都将匹配。您还可以通过使用havingValue
和matchIfMissing
属性创建更高级的检查。
如果在name
属性中给出了多个名称,则所有属性都必须通过测试才能使条件匹配。
Web 应用条件
@ConditionalOnWebApplication
和@ConditionalOnNotWebApplication
注解允许根据应用程序是否为 Web 应用程序来包含配置。基于 servlet 的 Web 应用程序是任何使用 Spring WebApplicationContext
、定义session
作用域或具有ConfigurableWebEnvironment
的应用程序。反应式 Web 应用程序是任何使用ReactiveWebApplicationContext
或具有ConfigurableReactiveWebEnvironment
的应用程序。
@ConditionalOnWarDeployment
和@ConditionalOnNotWarDeployment
注解允许根据应用程序是否为部署到 servlet 容器的传统 WAR 应用程序来包含配置。对于使用嵌入式 Web 服务器运行的应用程序,此条件将不匹配。
SpEL 表达式条件
@ConditionalOnExpression
注解允许根据SpEL 表达式的结果来包含配置。
在表达式中引用 bean 将导致该 bean 在上下文刷新处理的早期阶段被初始化。因此,该 bean 将不符合后处理(例如配置属性绑定)的条件,并且其状态可能不完整。 |
测试您的自动配置
自动配置可能会受到许多因素的影响:用户配置(@Bean
定义和Environment
自定义)、条件评估(特定库的存在)等。具体来说,每个测试都应该创建一个明确定义的ApplicationContext
,该上下文表示这些自定义的组合。ApplicationContextRunner
提供了一种实现此目标的好方法。
在本地镜像中运行测试时,ApplicationContextRunner 不起作用。 |
ApplicationContextRunner
通常定义为测试类的字段,以收集基本、通用的配置。以下示例确保始终调用MyServiceAutoConfiguration
-
Java
-
Kotlin
private final ApplicationContextRunner contextRunner = new ApplicationContextRunner()
.withConfiguration(AutoConfigurations.of(MyServiceAutoConfiguration.class));
val contextRunner = ApplicationContextRunner()
.withConfiguration(AutoConfigurations.of(MyServiceAutoConfiguration::class.java))
如果必须定义多个自动配置,则无需对其声明进行排序,因为它们将以与运行应用程序时完全相同的顺序调用。 |
每个测试都可以使用 runner 来表示特定的用例。例如,下面的示例调用用户配置(UserConfiguration
)并检查自动配置是否正确回退。调用run
提供了一个回调上下文,该上下文可以与AssertJ
一起使用。
-
Java
-
Kotlin
@Test
void defaultServiceBacksOff() {
this.contextRunner.withUserConfiguration(UserConfiguration.class).run((context) -> {
assertThat(context).hasSingleBean(MyService.class);
assertThat(context).getBean("myCustomService").isSameAs(context.getBean(MyService.class));
});
}
@Configuration(proxyBeanMethods = false)
static class UserConfiguration {
@Bean
MyService myCustomService() {
return new MyService("mine");
}
}
@Test
fun defaultServiceBacksOff() {
contextRunner.withUserConfiguration(UserConfiguration::class.java)
.run { context: AssertableApplicationContext ->
assertThat(context).hasSingleBean(MyService::class.java)
assertThat(context).getBean("myCustomService")
.isSameAs(context.getBean(MyService::class.java))
}
}
@Configuration(proxyBeanMethods = false)
internal class UserConfiguration {
@Bean
fun myCustomService(): MyService {
return MyService("mine")
}
}
还可以轻松自定义Environment
,如下例所示
-
Java
-
Kotlin
@Test
void serviceNameCanBeConfigured() {
this.contextRunner.withPropertyValues("user.name=test123").run((context) -> {
assertThat(context).hasSingleBean(MyService.class);
assertThat(context.getBean(MyService.class).getName()).isEqualTo("test123");
});
}
@Test
fun serviceNameCanBeConfigured() {
contextRunner.withPropertyValues("user.name=test123").run { context: AssertableApplicationContext ->
assertThat(context).hasSingleBean(MyService::class.java)
assertThat(context.getBean(MyService::class.java).name).isEqualTo("test123")
}
}
runner 还可以用于显示ConditionEvaluationReport
。可以在INFO
或DEBUG
级别打印报告。以下示例显示了如何使用ConditionEvaluationReportLoggingListener
在自动配置测试中打印报告。
-
Java
-
Kotlin
import org.junit.jupiter.api.Test;
import org.springframework.boot.autoconfigure.logging.ConditionEvaluationReportLoggingListener;
import org.springframework.boot.logging.LogLevel;
import org.springframework.boot.test.context.runner.ApplicationContextRunner;
class MyConditionEvaluationReportingTests {
@Test
void autoConfigTest() {
new ApplicationContextRunner()
.withInitializer(ConditionEvaluationReportLoggingListener.forLogLevel(LogLevel.INFO))
.run((context) -> {
// Test something...
});
}
}
import org.junit.jupiter.api.Test
import org.springframework.boot.autoconfigure.logging.ConditionEvaluationReportLoggingListener
import org.springframework.boot.logging.LogLevel
import org.springframework.boot.test.context.assertj.AssertableApplicationContext
import org.springframework.boot.test.context.runner.ApplicationContextRunner
class MyConditionEvaluationReportingTests {
@Test
fun autoConfigTest() {
ApplicationContextRunner()
.withInitializer(ConditionEvaluationReportLoggingListener.forLogLevel(LogLevel.INFO))
.run { context: AssertableApplicationContext? -> }
}
}
模拟 Web 上下文
如果需要测试仅在 servlet 或反应式 Web 应用程序上下文中运行的自动配置,请分别使用WebApplicationContextRunner
或ReactiveWebApplicationContextRunner
。
覆盖类路径
还可以测试在运行时特定类和/或包不存在时会发生什么情况。Spring Boot 附带了一个FilteredClassLoader
,runner 可以轻松使用它。在以下示例中,我们断言如果MyService
不存在,则自动配置将被正确禁用
-
Java
-
Kotlin
@Test
void serviceIsIgnoredIfLibraryIsNotPresent() {
this.contextRunner.withClassLoader(new FilteredClassLoader(MyService.class))
.run((context) -> assertThat(context).doesNotHaveBean("myService"));
}
@Test
fun serviceIsIgnoredIfLibraryIsNotPresent() {
contextRunner.withClassLoader(FilteredClassLoader(MyService::class.java))
.run { context: AssertableApplicationContext? ->
assertThat(context).doesNotHaveBean("myService")
}
}
创建您自己的启动器
一个典型的 Spring Boot 启动器包含用于自动配置和自定义给定技术(我们称之为“acme”)基础设施的代码。为了使其易于扩展,可以在专用命名空间中公开许多配置键到环境中。最后,提供了一个单一的“启动器”依赖项,以帮助用户尽可能轻松地入门。
具体来说,自定义启动器可以包含以下内容
-
包含“acme”自动配置代码的
autoconfigure
模块。 -
starter
模块,它提供对autoconfigure
模块以及“acme”和任何其他通常有用的依赖项的依赖关系。简而言之,添加启动器应该提供开始使用该库所需的一切。
这两个模块的分离绝不是必需的。如果“acme”有几种风格、选项或可选功能,那么最好将自动配置分开,因为您可以清楚地表达某些功能是可选的。此外,您可以创建提供对这些可选依赖项的意见的启动器。同时,其他人可以只依赖autoconfigure
模块,并使用不同的意见创建自己的启动器。
如果自动配置相对简单且没有可选功能,则将这两个模块合并到启动器中绝对是一个选项。
命名
您应该确保为您的启动器提供一个合适的命名空间。即使您使用不同的 Maven groupId
,也不要以spring-boot
开头您的模块名称。我们将来可能会为您自动配置的内容提供官方支持。
根据经验法则,您应该根据启动器命名组合模块。例如,假设您正在为“acme”创建启动器,并且您将自动配置模块命名为acme-spring-boot
,并将启动器命名为acme-spring-boot-starter
。如果您只有一个模块组合了这两个模块,则将其命名为acme-spring-boot-starter
。
配置键
如果您的启动器提供配置键,请为其使用唯一的命名空间。特别是,不要将您的键包含在 Spring Boot 使用的命名空间中(例如server
、management
、spring
等)。如果您使用相同的命名空间,我们将来可能会以破坏您的模块的方式修改这些命名空间。根据经验法则,以您拥有的命名空间(例如acme
)作为所有键的前缀。
确保通过为每个属性添加字段 Javadoc 来记录配置键,如下例所示
-
Java
-
Kotlin
import java.time.Duration;
import org.springframework.boot.context.properties.ConfigurationProperties;
@ConfigurationProperties("acme")
public class AcmeProperties {
/**
* Whether to check the location of acme resources.
*/
private boolean checkLocation = true;
/**
* Timeout for establishing a connection to the acme server.
*/
private Duration loginTimeout = Duration.ofSeconds(3);
// getters/setters ...
public boolean isCheckLocation() {
return this.checkLocation;
}
public void setCheckLocation(boolean checkLocation) {
this.checkLocation = checkLocation;
}
public Duration getLoginTimeout() {
return this.loginTimeout;
}
public void setLoginTimeout(Duration loginTimeout) {
this.loginTimeout = loginTimeout;
}
}
import org.springframework.boot.context.properties.ConfigurationProperties
import java.time.Duration
@ConfigurationProperties("acme")
class AcmeProperties(
/**
* Whether to check the location of acme resources.
*/
var isCheckLocation: Boolean = true,
/**
* Timeout for establishing a connection to the acme server.
*/
var loginTimeout:Duration = Duration.ofSeconds(3))
您应该只在@ConfigurationProperties 字段 Javadoc 中使用纯文本,因为在添加到 JSON 之前不会处理它们。 |
如果您使用记录类中的@ConfigurationProperties
,则应通过类级 Javadoc 标记@param
提供记录组件的描述(记录类中没有显式实例字段来放置常规字段级 Javadoc)。
以下是一些我们在内部遵循的规则,以确保描述的一致性
-
不要以“The”或“A”开头描述。
-
对于
boolean
类型,以“Whether”或“Enable”开头描述。 -
对于基于集合的类型,以“Comma-separated list”(逗号分隔列表)开头描述。
-
使用
java.time.Duration
而不是long
,并在默认单位与毫秒不同时描述默认单位,例如“如果未指定持续时间后缀,则将使用秒”。 -
不要在描述中提供默认值,除非必须在运行时确定。
确保触发元数据生成,以便您的键也可使用 IDE 辅助功能。您可能需要查看生成的元数据(META-INF/spring-configuration-metadata.json
)以确保您的键已正确记录。在兼容的 IDE 中使用您自己的启动器也是验证元数据质量的好方法。
“autoconfigure”模块
autoconfigure
模块包含开始使用库所需的一切。它还可以包含配置键定义(例如@ConfigurationProperties
)和任何回调接口,这些接口可用于进一步自定义组件的初始化方式。
您应该将对库的依赖项标记为可选,以便您可以更轻松地在项目中包含autoconfigure 模块。如果您这样做,则不会提供库,并且默认情况下,Spring Boot 会退避。 |
Spring Boot 使用注释处理器在元数据文件(META-INF/spring-autoconfigure-metadata.properties
)中收集自动配置的条件。如果该文件存在,则将其用于急切地过滤不匹配的自动配置,这将提高启动时间。
使用 Maven 构建时,建议在包含自动配置的模块中添加以下依赖项
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-autoconfigure-processor</artifactId>
<optional>true</optional>
</dependency>
如果您已在应用程序中直接定义了自动配置,请确保配置spring-boot-maven-plugin
以防止repackage
目标将依赖项添加到 uber jar 中
<project>
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<configuration>
<excludes>
<exclude>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-autoconfigure-processor</artifactId>
</exclude>
</excludes>
</configuration>
</plugin>
</plugins>
</build>
</project>
使用 Gradle 时,应在annotationProcessor
配置中声明依赖项,如下例所示
dependencies {
annotationProcessor "org.springframework.boot:spring-boot-autoconfigure-processor"
}
启动器模块
启动器实际上是一个空的 jar。其唯一目的是提供与库一起工作所需的依赖项。您可以将其视为开始所需内容的意见化视图。
不要对添加启动器的项目做出假设。如果要自动配置的库通常需要其他启动器,也请提及它们。如果可选依赖项的数量很多,则提供一组合适的默认依赖项可能很困难,因为您应该避免包含对库的典型使用不必要的依赖项。换句话说,您不应该包含可选依赖项。
无论哪种方式,您的启动器都必须直接或间接引用核心 Spring Boot 启动器(spring-boot-starter )(如果您的启动器依赖于另一个启动器,则无需添加它)。如果仅使用您的自定义启动器创建项目,则核心启动器的存在将使 Spring Boot 的核心功能得到尊重。 |