核心功能
1. SpringApplication 应用程序
该类提供了一种便捷的方法来引导从方法启动的 Spring 应用程序。
在许多情况下,您可以委托给 static 方法,如以下示例所示:SpringApplication
main()
SpringApplication.run
@SpringBootApplication
public class MyApplication {
public static void main(String[] args) {
SpringApplication.run(MyApplication.class, args);
}
}
@SpringBootApplication
class MyApplication
fun main(args: Array<String>) {
runApplication<MyApplication>(*args)
}
当您的应用程序启动时,您应该会看到类似于以下输出的内容:
. ____ _ __ _ _ /\\ / ___'_ __ _ _(_)_ __ __ _ \ \ \ \ ( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \ \\/ ___)| |_)| | | | | || (_| | ) ) ) ) ' |____| .__|_| |_|_| |_\__, | / / / / =========|_|==============|___/=/_/_/_/ :: Spring Boot :: (v3.2.11-SNAPSHOT) 2024-10-04T13:32:36.768Z INFO 83535 --- [ main] o.s.b.d.f.logexample.MyApplication : Starting MyApplication using Java 17.0.12 with PID 83535 (/opt/apps/myapp.jar started by myuser in /opt/apps/) 2024-10-04T13:32:36.776Z INFO 83535 --- [ main] o.s.b.d.f.logexample.MyApplication : No active profile set, falling back to 1 default profile: "default" 2024-10-04T13:32:39.246Z INFO 83535 --- [ main] o.s.b.w.embedded.tomcat.TomcatWebServer : Tomcat initialized with port 8080 (http) 2024-10-04T13:32:39.261Z INFO 83535 --- [ main] o.apache.catalina.core.StandardService : Starting service [Tomcat] 2024-10-04T13:32:39.262Z INFO 83535 --- [ main] o.apache.catalina.core.StandardEngine : Starting Servlet engine: [Apache Tomcat/10.1.30] 2024-10-04T13:32:39.358Z INFO 83535 --- [ main] o.a.c.c.C.[Tomcat].[localhost].[/] : Initializing Spring embedded WebApplicationContext 2024-10-04T13:32:39.359Z INFO 83535 --- [ main] w.s.c.ServletWebServerApplicationContext : Root WebApplicationContext: initialization completed in 2425 ms 2024-10-04T13:32:40.212Z INFO 83535 --- [ main] o.s.b.w.embedded.tomcat.TomcatWebServer : Tomcat started on port 8080 (http) with context path '' 2024-10-04T13:32:40.227Z INFO 83535 --- [ main] o.s.b.d.f.logexample.MyApplication : Started MyApplication in 4.469 seconds (process running for 5.308)
默认情况下,将显示日志记录消息,包括一些相关的启动详细信息,例如启动应用程序的用户。
如果您需要的日志级别不是 ,则可以设置它,如 Log Levels 中所述。
应用程序版本是使用主应用程序类的包中的实现版本确定的。
可以通过设置为 来关闭启动信息记录。
这还将关闭应用程序活动配置文件的日志记录。INFO
INFO
spring.main.log-startup-info
false
要在启动期间添加其他日志记录,可以在 的子类中覆盖。logStartupInfo(boolean) SpringApplication |
1.1. 启动失败
如果您的应用程序无法启动,注册后将有机会提供专用错误消息和解决问题的具体操作。
例如,如果您在 port 上启动一个 Web 应用程序,并且该端口已在使用中,您应该会看到类似于以下消息的内容:FailureAnalyzers
8080
*************************** APPLICATION FAILED TO START *************************** Description: Embedded servlet container failed to start. Port 8080 was already in use. Action: Identify and stop the process that is listening on port 8080 or configure this application to listen on another port.
Spring Boot 提供了许多实现,您可以添加自己的实现。FailureAnalyzer |
如果没有故障分析器能够处理异常,您仍然可以显示完整的条件报告,以更好地了解出了什么问题。
为此,您需要启用 debug
属性或为 .
org.springframework.boot.autoconfigure.logging.ConditionEvaluationReportLoggingListener
例如,如果您使用 运行应用程序,则可以按如下方式启用该属性:java -jar
debug
$ java -jar myproject-0.0.1-SNAPSHOT.jar --debug
1.2. 延迟初始化
SpringApplication
允许延迟初始化应用程序。
启用延迟初始化后,将根据需要创建 bean,而不是在应用程序启动期间创建 bean。
因此,启用延迟初始化可以减少应用程序启动所需的时间。
在 Web 应用程序中,启用延迟初始化将导致许多与 Web 相关的 bean 在收到 HTTP 请求之前不会初始化。
延迟初始化的一个缺点是,它可能会延迟发现应用程序的问题。 如果延迟初始化了配置错误的 bean,则在启动期间将不再发生失败,并且只有在初始化 bean 时问题才会变得明显。 还必须注意确保 JVM 具有足够的内存来容纳应用程序的所有 bean,而不仅仅是在启动期间初始化的 bean。 由于这些原因,默认情况下不启用延迟初始化,建议在启用延迟初始化之前对 JVM 的堆大小进行微调。
可以使用 on 或 on 方法以编程方式启用延迟初始化。
或者,可以使用以下示例中所示的属性启用它:lazyInitialization
SpringApplicationBuilder
setLazyInitialization
SpringApplication
spring.main.lazy-initialization
spring.main.lazy-initialization=true
spring:
main:
lazy-initialization: true
如果要对某些 bean 禁用延迟初始化,同时对应用程序的其余部分使用延迟初始化,则可以使用 Comments 将其 lazy 属性显式设置为 false。@Lazy(false) |
1.3. 自定义横幅
可以通过将文件添加到 Classpath 或将属性设置为此类文件的位置来更改启动时打印的标题。
如果文件的编码不是 UTF-8,则可以设置 .banner.txt
spring.banner.location
spring.banner.charset
在文件中,您可以使用 中的任何可用键以及以下任何占位符:banner.txt
Environment
变量 | 描述 |
---|---|
|
应用程序中声明的应用程序版本号 。
例如,打印为 。 |
|
应用程序的版本号,在 中声明并格式化为显示(用括号括起来,前缀为 )。
例如。 |
|
您正在使用的 Spring Boot 版本。
例如。 |
|
您正在使用的 Spring Boot 版本,格式化为显示(用括号括起来,前缀为 )。
例如。 |
|
其中 是 ANSI 转义码的名称。
有关详细信息,请参阅 |
|
应用程序中声明的应用程序标题。
例如,打印为 . |
如果要以编程方式生成横幅,可以使用该方法。
使用该接口并实现您自己的方法。SpringApplication.setBanner(…) org.springframework.boot.Banner printBanner() |
您还可以使用该属性来确定横幅是否必须打印在 () 上、发送到配置的记录器 () 或根本不生成 ()。spring.main.banner-mode
System.out
console
log
off
打印的横幅将注册为以下名称下的单例 Bean: 。springBootBanner
这些 , 和 属性仅在您使用 Spring Boot Starters或与之一起使用时可用。
如果您运行的是解压缩的 jar 并使用本机映像启动它或将应用程序作为本机映像运行,则不会解析这些值。 要使用这些属性,请使用 |
1.4. 自定义 SpringApplication
如果默认值不符合您的口味,您可以改为创建本地实例并对其进行自定义。
例如,要关闭横幅,您可以编写:SpringApplication
@SpringBootApplication
public class MyApplication {
public static void main(String[] args) {
SpringApplication application = new SpringApplication(MyApplication.class);
application.setBannerMode(Banner.Mode.OFF);
application.run(args);
}
}
@SpringBootApplication
class MyApplication
fun main(args: Array<String>) {
runApplication<MyApplication>(*args) {
setBannerMode(Banner.Mode.OFF)
}
}
传递给的构造函数参数是 Spring bean 的配置源。
在大多数情况下,这些是对类的引用,但它们也可以是直接引用类。SpringApplication @Configuration @Component |
也可以使用文件进行配置。
有关详细信息,请参阅 外部化配置 。SpringApplication
application.properties
有关配置选项的完整列表,请参阅 SpringApplication
Javadoc。
1.5. Fluent Builder API
如果您需要构建层次结构(具有父/子关系的多个上下文),或者您更喜欢使用“Fluent”构建器 API,则可以使用 .ApplicationContext
SpringApplicationBuilder
这允许您将多个方法调用和包含以及允许您创建层次结构的方法链接在一起,如以下示例所示:SpringApplicationBuilder
parent
child
new SpringApplicationBuilder().sources(Parent.class)
.child(Application.class)
.bannerMode(Banner.Mode.OFF)
.run(args);
SpringApplicationBuilder()
.sources(Parent::class.java)
.child(Application::class.java)
.bannerMode(Banner.Mode.OFF)
.run(*args)
创建层次结构时存在一些限制。
例如,Web 组件必须包含在子上下文中,并且父上下文和子上下文都使用相同的 Web 组件。
有关完整详细信息,请参见 SpringApplicationBuilder Javadoc。ApplicationContext Environment |
1.6. 应用程序可用性
在平台上部署时,应用程序可以使用 Kubernetes 探针等基础设施向平台提供有关其可用性的信息。 Spring Boot 包括对常用的“liveness”和“readiness”可用性状态的开箱即用支持。 如果您使用的是 Spring Boot 的“actuator”支持,则这些状态将作为运行状况端点组公开。
此外,您还可以通过将接口注入到您自己的 bean 中来获取可用性状态。ApplicationAvailability
1.6.1. 活跃状态
应用程序的 “Liveness” 状态表明其内部状态是否允许它正常工作,或者如果当前出现故障,则自行恢复。 损坏的 “Liveness” 状态意味着应用程序处于无法恢复的状态,基础设施应重新启动应用程序。
通常,“Liveness” 状态不应基于外部检查,例如 Health 检查。 如果是这样,则失败的外部系统(数据库、Web API、外部缓存)将在整个平台上触发大规模重启和级联故障。 |
Spring Boot 应用程序的内部状态主要由 Spring 表示。
如果应用程序上下文已成功启动,则 Spring Boot 假定应用程序处于有效状态。
一旦刷新了上下文,应用程序就被视为活动了,请参阅 Spring Boot 应用程序生命周期和相关应用程序事件。ApplicationContext
1.6.2. 就绪状态
应用程序的 “Readiness” 状态表明应用程序是否已准备好处理流量。
失败的 “Readiness” 状态告诉平台它暂时不应将流量路由到应用程序。
这通常发生在启动期间、处理组件时,或者如果应用程序决定太忙而无法进行其他流量,则随时发生。CommandLineRunner
ApplicationRunner
一旦调用了应用程序和命令行运行器,应用程序就被视为准备就绪,请参阅 Spring Boot 应用程序生命周期和相关应用程序事件。
预期在启动期间运行的任务应由 and 组件执行,而不是使用 Spring 组件生命周期回调,例如 .CommandLineRunner ApplicationRunner @PostConstruct |
1.6.3. 管理应用程序可用性状态
应用程序组件可以随时通过注入接口并对其调用方法来检索当前可用性状态。
更常见的是,应用程序需要侦听状态更新或更新应用程序的状态。ApplicationAvailability
例如,我们可以将应用程序的 “Readiness” 状态导出到一个文件中,以便 Kubernetes “exec Probe” 可以查看此文件:
@Component
public class MyReadinessStateExporter {
@EventListener
public void onStateChange(AvailabilityChangeEvent<ReadinessState> event) {
switch (event.getState()) {
case ACCEPTING_TRAFFIC -> {
// create file /tmp/healthy
}
case REFUSING_TRAFFIC -> {
// remove file /tmp/healthy
}
}
}
}
@Component
class MyReadinessStateExporter {
@EventListener
fun onStateChange(event: AvailabilityChangeEvent<ReadinessState?>) {
when (event.state) {
ReadinessState.ACCEPTING_TRAFFIC -> {
// create file /tmp/healthy
}
ReadinessState.REFUSING_TRAFFIC -> {
// remove file /tmp/healthy
}
else -> {
// ...
}
}
}
}
当应用程序中断且无法恢复时,我们还可以更新应用程序的状态:
@Component
public class MyLocalCacheVerifier {
private final ApplicationEventPublisher eventPublisher;
public MyLocalCacheVerifier(ApplicationEventPublisher eventPublisher) {
this.eventPublisher = eventPublisher;
}
public void checkLocalCache() {
try {
// ...
}
catch (CacheCompletelyBrokenException ex) {
AvailabilityChangeEvent.publish(this.eventPublisher, ex, LivenessState.BROKEN);
}
}
}
@Component
class MyLocalCacheVerifier(private val eventPublisher: ApplicationEventPublisher) {
fun checkLocalCache() {
try {
// ...
} catch (ex: CacheCompletelyBrokenException) {
AvailabilityChangeEvent.publish(eventPublisher, ex, LivenessState.BROKEN)
}
}
}
1.7. 应用程序事件和侦听器
除了通常的 Spring Framework 事件(例如ContextRefreshedEvent
)之外,a 还会发送一些其他应用程序事件。SpringApplication
某些事件实际上是在创建 之前触发的,因此您无法在这些事件上将侦听器注册为 。
您可以使用 method 或 method 注册它们。 如果您希望自动注册这些侦听器,而不管应用程序是如何创建的,您都可以将文件添加到项目中并使用 key 引用您的侦听器,如以下示例所示: org.springframework.context.ApplicationListener=com.example.project.MyListener |
应用程序事件在应用程序运行时按以下顺序发送:
-
An 在运行开始时发送,但在任何处理之前发送,侦听器和初始值设定项的注册除外。
ApplicationStartingEvent
-
当 要在上下文中使用的 已知但在创建上下文之前,将发送 An。
ApplicationEnvironmentPreparedEvent
Environment
-
当准备好并调用ApplicationContextInitializers时,但在加载任何 bean 定义之前,将发送一个。
ApplicationContextInitializedEvent
ApplicationContext
-
在刷新开始之前,但在加载 Bean 定义之后发送 AN。
ApplicationPreparedEvent
-
在刷新上下文之后,但在调用任何应用程序和命令行运行程序之前发送 AN。
ApplicationStartedEvent
-
An 紧随其后发送 with,以指示应用程序被视为实时应用程序。
AvailabilityChangeEvent
LivenessState.CORRECT
-
在调用任何应用程序和命令行运行程序后发送 AN。
ApplicationReadyEvent
-
An 紧随其后发送 with,以指示应用程序已准备好为请求提供服务。
AvailabilityChangeEvent
ReadinessState.ACCEPTING_TRAFFIC
-
如果启动时出现异常,则发送 An。
ApplicationFailedEvent
上面的列表仅包含绑定到 .
除此之外,以下事件还会在 之后 和 之前发布 :SpringApplicationEvent
SpringApplication
ApplicationPreparedEvent
ApplicationStartedEvent
-
A 在 准备就绪后发送。 和 分别是 servlet 和 reactive 变体。
WebServerInitializedEvent
WebServer
ServletWebServerInitializedEvent
ReactiveWebServerInitializedEvent
-
刷新 时发送 A。
ContextRefreshedEvent
ApplicationContext
您通常不需要使用应用程序事件,但知道它们存在会很方便。 在内部, Spring Boot 使用事件来处理各种任务。 |
事件侦听器不应运行可能很长的任务,因为它们默认在同一线程中执行。 请考虑改用应用程序和命令行运行程序。 |
应用程序事件是使用 Spring Framework 的事件发布机制发送的。
此机制的一部分可确保发布到子上下文中的侦听器的事件也发布到任何祖先上下文中的侦听器。
因此,如果您的应用程序使用实例层次结构,则侦听器可能会接收相同类型应用程序事件的多个实例。SpringApplication
要使侦听器能够区分其上下文的事件和后代上下文的事件,它应请求注入其应用程序上下文,然后将注入的上下文与事件的上下文进行比较。
可以通过实现来注入上下文,或者,如果侦听器是 bean,则通过使用 。ApplicationContextAware
@Autowired
1.8. Web 环境
A 尝试代表您创建正确的 类型。
用于确定 a 的算法如下:SpringApplication
ApplicationContext
WebApplicationType
-
如果存在 Spring MVC,则使用
AnnotationConfigServletWebServerApplicationContext
-
如果 Spring MVC 不存在而 Spring WebFlux 存在,则使用
AnnotationConfigReactiveWebServerApplicationContext
-
否则,使用
AnnotationConfigApplicationContext
这意味着,如果你在同一个应用程序中使用 Spring MVC 和 Spring WebFlux 中的新版本,则默认情况下将使用 Spring MVC。
您可以通过调用 .WebClient
setWebApplicationType(WebApplicationType)
还可以通过调用 .ApplicationContext
setApplicationContextFactory(…)
在 JUnit 测试中使用时,通常需要调用 Call。setWebApplicationType(WebApplicationType.NONE) SpringApplication |
1.9. 访问应用程序参数
如果需要访问传递给 的应用程序参数,则可以注入一个 bean。
该接口提供对原始参数以及 parsed 和 arguments 的访问,如以下示例所示:SpringApplication.run(…)
org.springframework.boot.ApplicationArguments
ApplicationArguments
String[]
option
non-option
@Component
public class MyBean {
public MyBean(ApplicationArguments args) {
boolean debug = args.containsOption("debug");
List<String> files = args.getNonOptionArgs();
if (debug) {
System.out.println(files);
}
// if run with "--debug logfile.txt" prints ["logfile.txt"]
}
}
@Component
class MyBean(args: ApplicationArguments) {
init {
val debug = args.containsOption("debug")
val files = args.nonOptionArgs
if (debug) {
println(files)
}
// if run with "--debug logfile.txt" prints ["logfile.txt"]
}
}
Spring Boot 还向 Spring 注册 a 。
这还允许您使用 annotation 注入单个应用程序参数。CommandLinePropertySource Environment @Value |
1.10. 使用 ApplicationRunner 或 CommandLineRunner
如果你需要在启动后运行一些特定的代码,你可以实现 or 接口。
这两个接口的工作方式相同,并提供单个方法,该方法在完成之前调用。SpringApplication
ApplicationRunner
CommandLineRunner
run
SpringApplication.run(…)
此协定非常适合应在应用程序启动之后但在开始接受流量之前运行的任务。 |
接口以字符串数组的形式提供对应用程序参数的访问,而 则使用前面讨论的接口。
以下示例显示了 with a 方法:CommandLineRunner
ApplicationRunner
ApplicationArguments
CommandLineRunner
run
@Component
public class MyCommandLineRunner implements CommandLineRunner {
@Override
public void run(String... args) {
// Do something...
}
}
@Component
class MyCommandLineRunner : CommandLineRunner {
override fun run(vararg args: String) {
// Do something...
}
}
如果定义了多个 或 bean,并且必须按特定 Sequences 调用,则还可以实现接口或使用 Comments。CommandLineRunner
ApplicationRunner
org.springframework.core.Ordered
org.springframework.core.annotation.Order
1.11. 应用程序退出
每个 API 都会向 JVM 注册一个 shutdown 钩子,以确保 在退出时正常关闭。
可以使用所有标准的 Spring 生命周期回调(例如接口或 Comments)。SpringApplication
ApplicationContext
DisposableBean
@PreDestroy
此外,如果 bean 希望在调用时返回特定的退出代码,则可以实现该接口。
然后,可以将此退出代码传递给 以将其作为状态代码返回,如以下示例所示:org.springframework.boot.ExitCodeGenerator
SpringApplication.exit()
System.exit()
@SpringBootApplication
public class MyApplication {
@Bean
public ExitCodeGenerator exitCodeGenerator() {
return () -> 42;
}
public static void main(String[] args) {
System.exit(SpringApplication.exit(SpringApplication.run(MyApplication.class, args)));
}
}
@SpringBootApplication
class MyApplication {
@Bean
fun exitCodeGenerator() = ExitCodeGenerator { 42 }
}
fun main(args: Array<String>) {
exitProcess(SpringApplication.exit(
runApplication<MyApplication>(*args)))
}
此外,该接口可能由异常实现。
当遇到此类异常时, Spring Boot 返回已实现的方法提供的退出代码。ExitCodeGenerator
getExitCode()
如果有多个 ,则使用生成的第一个非零退出代码。
要控制生成器的调用顺序,请另外实现接口或使用 Comments。ExitCodeGenerator
org.springframework.core.Ordered
org.springframework.core.annotation.Order
1.12. 管理员功能
可以通过指定属性为应用程序启用与管理员相关的功能。
这会在 platform 上公开 SpringApplicationAdminMXBean
。
您可以使用此功能远程管理 Spring Boot 应用程序。
此功能对于任何服务包装器实现也很有用。spring.application.admin.enabled
MBeanServer
如果您想知道应用程序正在哪个 HTTP 端口上运行,请获取键为 .local.server.port |
1.13. 应用程序启动跟踪
在应用程序启动期间,和 执行许多与应用程序生命周期相关的任务,
bean 生命周期,甚至处理应用程序事件。
通过ApplicationStartup
, Spring Framework 允许你使用StartupStep
对象跟踪应用程序启动序列。
收集此数据可以用于分析目的,或者只是为了更好地了解应用程序启动过程。SpringApplication
ApplicationContext
您可以在设置实例时选择实施。
例如,要使用 ,您可以编写:ApplicationStartup
SpringApplication
BufferingApplicationStartup
@SpringBootApplication
public class MyApplication {
public static void main(String[] args) {
SpringApplication application = new SpringApplication(MyApplication.class);
application.setApplicationStartup(new BufferingApplicationStartup(2048));
application.run(args);
}
}
@SpringBootApplication
class MyApplication
fun main(args: Array<String>) {
runApplication<MyApplication>(*args) {
applicationStartup = BufferingApplicationStartup(2048)
}
}
第一个可用的实现由 Spring Framework 提供。
它将特定于 Spring 的启动事件添加到 Java Flight Recorder 会话中,用于分析应用程序并将其 Spring 上下文生命周期与 JVM 事件(例如分配、GC、类加载等)相关联。
配置后,您可以通过在启用 Flight Recorder 的情况下运行应用程序来记录数据:FlightRecorderApplicationStartup
$ java -XX:StartFlightRecording:filename=recording.jfr,duration=10s -jar demo.jar
Spring Boot 附带了该变体;此实现用于缓冲启动步骤并将其排空到外部 Metrics 系统中。
应用程序可以在任何组件中请求 type 为 的 bean。BufferingApplicationStartup
BufferingApplicationStartup
Spring Boot 还可以配置为公开一个启动
端点,该端点将此信息作为 JSON 文档提供。
1.14. 虚拟线程
如果您在 Java 21 或更高版本上运行,则可以通过将属性设置为 .spring.threads.virtual.enabled
true
在为您的应用程序启用此选项之前,您应该考虑阅读官方 Java 虚拟线程文档。
在某些情况下,由于“固定虚拟线程”,应用程序的吞吐量可能会降低;本页还介绍了如何使用 JDK Flight Recorder 或 CLI 检测此类情况。jcmd
如果启用了虚拟线程,则配置线程池的属性将不再有效。 这是因为虚拟线程被调度在 JVM 范围的平台线程池上,而不是在专用线程池上。 |
虚拟线程的一个副作用是它们是守护程序线程。
如果 JVM 的所有线程都是守护程序线程,则 JVM 将退出。
例如,当您依赖 bean 来保持应用程序活动状态时,此行为可能会成为一个问题。
如果使用虚拟线程,则调度程序线程是虚拟线程,因此是守护程序线程,不会使 JVM 保持活动状态。
这不仅会影响调度,其他技术也可能如此。
要使 JVM 在所有情况下都运行,建议将属性设置为 。
这可确保 JVM 保持活动状态,即使所有线程都是虚拟线程也是如此。@Scheduled spring.main.keep-alive true |
2. 外部化配置
Spring Boot 允许您外部化配置,以便您可以在不同环境中使用相同的应用程序代码。 您可以使用各种外部配置源,包括 Java 属性文件、YAML 文件、环境变量和命令行参数。
属性值可以通过使用 Comments 直接注入到 bean 中,通过 Spring 的抽象访问,或者通过 .@Value
Environment
@ConfigurationProperties
Spring Boot 使用一个非常特殊的 Sequences,旨在允许合理地覆盖值。
后面的属性源可以覆盖前面的属性源中定义的值。
按以下顺序考虑源:PropertySource
-
默认属性(通过设置 指定)。
SpringApplication.setDefaultProperties
-
@PropertySource
类上的注释。 请注意,在刷新应用程序上下文之前,此类属性源不会添加到 。 现在配置某些属性(例如,在刷新开始之前读取的 and)为时已晚。@Configuration
Environment
logging.*
spring.main.*
-
配置数据(例如文件)。
application.properties
-
A 仅在 中具有属性。
RandomValuePropertySource
random.*
-
OS 环境变量。
-
Java 系统属性 ()。
System.getProperties()
-
JNDI 属性来自 .
java:comp/env
-
ServletContext
init 参数。 -
ServletConfig
init 参数。 -
属性(嵌入在环境变量或系统属性中的内联 JSON)。
SPRING_APPLICATION_JSON
-
命令行参数。
-
properties
属性。 可用于@SpringBootTest
和测试注释,用于测试应用程序的特定切片。 -
@DynamicPropertySource
Comments。 -
@TestPropertySource
测试中的注释。 -
devtools 处于活动状态时目录中的 devtools 全局设置属性。
$HOME/.config/spring-boot
配置数据文件按以下顺序考虑:
-
打包在 jar 中的应用程序属性(和 YAML 变体)。
application.properties
-
打包在 jar 中(和 YAML 变体)的特定于配置文件的应用程序属性。
application-{profile}.properties
-
打包的 jar(和 YAML 变体)之外的应用程序属性。
application.properties
-
打包的 jar(和 YAML 变体)之外特定于配置文件的应用程序属性。
application-{profile}.properties
建议整个应用程序坚持使用一种格式。
如果在同一位置有 和 YAML 格式的配置文件,则优先。.properties .properties |
如果使用环境变量而不是系统属性,则大多数操作系统不允许使用句点分隔的键名称,但您可以改用下划线(例如,而不是 )。
有关详细信息,请参阅从环境变量绑定。SPRING_CONFIG_NAME spring.config.name |
如果应用程序在 servlet 容器或应用程序服务器中运行,则可以使用 JNDI 属性 (in) 或 servlet 上下文初始化参数来代替环境变量或系统属性。java:comp/env |
为了提供具体示例,假设您开发了一个使用属性的 a,如以下示例所示:@Component
name
@Component
public class MyBean {
@Value("${name}")
private String name;
// ...
}
@Component
class MyBean {
@Value("\${name}")
private val name: String? = null
// ...
}
在应用程序类路径上(例如,在 jar 中),您可以有一个文件,该文件为 .
在新环境中运行时,可以在 jar 外部提供一个文件,该文件会覆盖 .
对于一次性测试,您可以使用特定的命令行开关(例如)。application.properties
name
application.properties
name
java -jar app.jar --name="Spring"
和 endpoints 可用于确定属性具有特定值的原因。
您可以使用这两个终端节点来诊断意外的属性值。
有关详细信息,请参阅“生产就绪功能”部分。env configprops |
2.1. 访问命令行属性
默认情况下,将任何命令行选项参数(即以 、 开头的参数,例如 )转换为 a 并将它们添加到 Spring 中。
如前所述,命令行属性始终优先于基于文件的属性源。SpringApplication
--
--server.port=9000
property
Environment
如果不希望将命令行属性添加到 中,则可以使用 来禁用它们。Environment
SpringApplication.setAddCommandLineProperties(false)
2.2. JSON 应用程序属性
环境变量和系统属性通常具有限制,这意味着无法使用某些属性名称。 为了帮助解决这个问题, Spring Boot 允许您将一个属性块编码为单个 JSON 结构。
当您的应用程序启动时,将解析 any 或 属性并将其添加到 .spring.application.json
SPRING_APPLICATION_JSON
Environment
例如,可以在 UN*X shell 的命令行中将该属性作为环境变量提供:SPRING_APPLICATION_JSON
$ SPRING_APPLICATION_JSON='{"my":{"name":"test"}}' java -jar myapp.jar
在前面的示例中,您最终会在 Spring .my.name=test
Environment
相同的 JSON 也可以作为系统属性提供:
$ java -Dspring.application.json='{"my":{"name":"test"}}' -jar myapp.jar
或者,您可以使用命令行参数提供 JSON:
$ java -jar myapp.jar --spring.application.json='{"my":{"name":"test"}}'
如果要部署到经典 Application Server,则还可以使用名为 的 JNDI 变量。java:comp/env/spring.application.json
尽管 JSON 中的值将添加到生成的属性源中,但 会将属性视为缺失值。
这意味着 JSON 无法使用值覆盖低阶属性源中的属性。null PropertySourcesPropertyResolver null null |
2.3. 外部应用程序属性
Spring Boot 将在应用程序启动时自动从以下位置查找并加载文件:application.properties
application.yaml
-
从 classpath
-
类路径根
-
classpath 包
/config
-
-
从当前目录
-
当前目录
-
当前目录中的子目录
config/
-
子目录的直接子目录
config/
-
该列表按优先级排序(较低项中的值将覆盖较早项的值)。
加载文件中的文档将添加到 Spring 中。PropertySources
Environment
如果您不喜欢作为配置文件名,则可以通过指定 environment 属性来切换到其他文件名。
例如,要查找 and 文件,您可以按如下方式运行应用程序:application
spring.config.name
myproject.properties
myproject.yaml
$ java -jar myproject.jar --spring.config.name=myproject
您还可以使用 environment 属性引用显式位置。
此属性接受要检查的一个或多个位置的逗号分隔列表。spring.config.location
以下示例显示如何指定两个不同的文件:
$ java -jar myproject.jar --spring.config.location=\
optional:classpath:/default.properties,\
optional:classpath:/override.properties
如果位置是可选的,并且您不介意它们不存在,请使用前缀。optional: |
spring.config.name 、 和 用于确定必须加载哪些文件。
它们必须定义为环境属性(通常是 OS 环境变量、系统属性或命令行参数)。spring.config.location spring.config.additional-location |
如果包含目录(而不是文件),则它们应以 .
在运行时,它们将附加加载之前生成的名称。
中指定的文件将直接导入。spring.config.location
/
spring.config.name
spring.config.location
目录和文件位置值也都进行了扩展,以检查特定于配置文件的文件。
例如,如果您有 of ,您还会发现加载了相应的文件。spring.config.location classpath:myconfig.properties classpath:myconfig-<profile>.properties |
在大多数情况下,您添加的每个项目都将引用一个文件或目录。
位置按其定义顺序进行处理,较晚的位置可以覆盖较早的位置的值。spring.config.location
如果你有一个复杂的位置设置,并且你使用特定于配置文件的配置文件,你可能需要提供进一步的提示,以便 Spring Boot 知道应该如何对它们进行分组。
位置组是所有位置都被视为同一级别的位置的集合。
例如,您可能希望对所有 Classpath 位置进行分组,然后对所有外部位置进行分组。
位置组中的项目应以 分隔。
有关更多详细信息,请参阅“分析特定文件”部分中的示例。;
使用 配置的位置将替换默认位置。
例如,如果配置了值 ,则考虑的完整位置集为:spring.config.location
spring.config.location
optional:classpath:/custom-config/,optional:file:./custom-config/
-
optional:classpath:custom-config/
-
optional:file:./custom-config/
如果您希望添加其他位置而不是替换它们,则可以使用 .
从其他位置加载的属性可以覆盖默认位置中的属性。
例如,如果配置了值 ,则考虑的完整位置集为:spring.config.additional-location
spring.config.additional-location
optional:classpath:/custom-config/,optional:file:./custom-config/
-
optional:classpath:/;optional:classpath:/config/
-
optional:file:./;optional:file:./config/;optional:file:./config/*/
-
optional:classpath:custom-config/
-
optional:file:./custom-config/
此搜索顺序允许您在一个配置文件中指定默认值,然后在另一个配置文件中选择性地覆盖这些值。
您可以在其中一个默认位置 (或您选择的任何其他 basename ) 中为应用程序提供默认值。
然后,可以在运行时使用位于其中一个自定义位置的不同文件覆盖这些默认值。application.properties
spring.config.name
2.3.1. 可选位置
默认情况下,当指定的 config 数据位置不存在时, Spring Boot 将抛出一个,并且您的应用程序将不会启动。ConfigDataLocationNotFoundException
如果要指定位置,但不介意它并不总是存在,则可以使用前缀。
你可以将此前缀与and属性以及spring.config.import
声明一起使用。optional:
spring.config.location
spring.config.additional-location
例如,值 of 允许您的应用程序启动,即使文件缺失也是如此。spring.config.import
optional:file:./myconfig.properties
myconfig.properties
如果要忽略所有并始终继续启动应用程序,则可以使用该属性。
将值设置为 using 或 with a system/environment variable。ConfigDataLocationNotFoundExceptions
spring.config.on-not-found
ignore
SpringApplication.setDefaultProperties(…)
2.3.2. 通配符位置
如果配置文件位置包含最后一个路径段的字符,则将其视为通配符位置。
加载配置时,通配符会展开,以便同时检查直接子目录。
通配符位置在 Kubernetes 等环境中,当有多个 config 属性来源时特别有用。*
例如,如果您有一些 Redis 配置和一些 MySQL 配置,您可能希望将这两个配置分开,同时要求这两个配置都存在于一个文件中。
这可能会导致两个单独的文件挂载在不同的位置,例如 和 。
在这种情况下,如果通配符位置为 ,将导致两个文件都被处理。application.properties
application.properties
/config/redis/application.properties
/config/mysql/application.properties
config/*/
默认情况下, Spring Boot 包含在默认搜索位置中。
这意味着将搜索 jar 之外的目录的所有子目录。config/*/
/config
您可以自己将通配符位置与 和 属性一起使用。spring.config.location
spring.config.additional-location
通配符位置必须仅包含一个通配符位置,对于作为目录的搜索位置或作为文件的搜索位置,通配符位置必须以 结尾。
带有通配符的位置根据文件名的绝对路径按字母顺序排序。* */ */<filename> |
通配符位置仅适用于外部目录。
不能在某个位置使用通配符。classpath: |
2.3.3. 分析特定文件
除了属性文件,Spring Boot 还将尝试使用 naming convention 加载特定于配置文件的文件。
例如,如果您的应用程序激活名为 YAML 文件的配置文件并使用 YAML 文件,则 和 都将被考虑。application
application-{profile}
prod
application.yaml
application-prod.yaml
特定于配置文件的属性从与 standard 相同的位置加载,特定于配置文件的文件始终覆盖非特定文件。
如果指定了多个配置文件,则适用 last-wins 策略。
例如,如果配置文件由属性指定,则 中的值可以被 中的值覆盖。application.properties
prod,live
spring.profiles.active
application-prod.properties
application-live.properties
“最后获胜”策略适用于营业地点组级别。
A of 将具有与 . 例如,继续上面的示例,我们可能有以下文件: /cfg application-live.properties /ext application-live.properties application-prod.properties 当我们有 a of 时,我们会先处理所有文件,然后再处理所有文件:
当我们有 (带有分隔符) 时,我们会在同一级别处理 and:
|
具有一组默认配置文件(默认情况下 ),如果未设置活动配置文件,则使用这些配置文件。
换句话说,如果未明确激活任何用户档案,则考虑 中的属性。Environment
[default]
application-default
属性文件只加载一次。 如果您已经直接导入了特定于配置文件的属性文件,则不会再次导入它。 |
2.3.4. 导入其他数据
应用程序属性可以使用该属性从其他位置导入进一步的配置数据。
导入在被发现时进行处理,并被视为插入到声明导入的导入文档的正下方的附加文档。spring.config.import
例如,您的 Classpath 文件中可能包含以下内容:application.properties
spring.application.name=myapp
spring.config.import=optional:file:./dev.properties
spring:
application:
name: "myapp"
config:
import: "optional:file:./dev.properties"
这将触发当前目录中文件的导入(如果存在此类文件)。
导入的值将优先于触发导入的文件。
在上面的示例中,可以重新定义为不同的值。dev.properties
dev.properties
dev.properties
spring.application.name
无论声明多少次,导入都只会导入一次。 在 properties/yaml 文件中的单个文档中定义导入的顺序无关紧要。 例如,下面的两个示例产生相同的结果:
spring.config.import=my.properties
my.property=value
spring:
config:
import: "my.properties"
my:
property: "value"
my.property=value
spring.config.import=my.properties
my:
property: "value"
spring:
config:
import: "my.properties"
在上述两个示例中,文件中的值将优先于触发其导入的文件。my.properties
可以在单个键下指定多个位置。
位置将按照定义的顺序进行处理,以后的导入优先。spring.config.import
在适当时,还会考虑导入特定于 Profile 的变体。
上面的示例将导入 both 以及 any variants。my.properties my-<profile>.properties |
Spring Boot 包含可插拔 API,允许支持各种不同的位置地址。 默认情况下,您可以导入 Java 属性、YAML 和“配置树”。 第三方 jar 可以提供对其他技术的支持(不要求文件是本地的)。 例如,您可以想象配置数据来自 Consul、Apache ZooKeeper 或 Netflix Archaius 等外部存储。 如果要支持自己的位置,请参阅包中的 和 类。 |
2.3.5. 导入无扩展文件
某些云平台无法向卷挂载的文件添加文件扩展名。 要导入这些无扩展名文件,你需要给 Spring Boot 一个提示,以便它知道如何加载它们。 您可以通过在方括号中放置扩展提示来执行此操作。
例如,假设您有一个希望作为 yaml 导入的文件。
您可以使用以下方法将其导入:/etc/config/myconfig
application.properties
spring.config.import=file:/etc/config/myconfig[.yaml]
spring:
config:
import: "file:/etc/config/myconfig[.yaml]"
2.3.6. 使用配置树
在云平台(如 Kubernetes)上运行应用程序时,您通常需要读取平台提供的配置值。 将环境变量用于此类目的并不少见,但这可能有缺点,尤其是在值应该保密的情况下。
可以使用两种常见的卷挂载模式:
-
单个文件包含一组完整的属性(通常写为 YAML)。
-
多个文件被写入目录树,文件名成为 'key',内容成为 'value'。
对于第一种情况,您可以如上所述直接使用 YAML 或 Properties 文件导入。
对于第二种情况,你需要使用前缀,以便 Spring Boot 知道它需要将所有文件作为属性公开。spring.config.import
configtree:
例如,假设 Kubernetes 挂载了以下卷:
etc/ config/ myapp/ username password
文件的内容将是 config 值,而 的内容将是 secret。username
password
要导入这些属性,您可以将以下内容添加到您的 or 文件中:application.properties
application.yaml
spring.config.import=optional:configtree:/etc/config/
spring:
config:
import: "optional:configtree:/etc/config/"
然后,你可以以通常的方式从 中访问或注入 和 属性。myapp.username
myapp.password
Environment
配置树下的文件夹和文件的名称构成属性名称。
在上面的示例中,要访问属性 as 和 ,您可以设置为 。username password spring.config.import optional:configtree:/etc/config/myapp |
带有点表示法的文件名也会正确映射。
例如,在上面的示例中,名为 in 的文件将导致 .myapp.username /etc/config myapp.username Environment |
配置树值可以绑定到 string 和 types,具体取决于预期的内容。String byte[] |
如果要从同一个父文件夹导入多个配置树,则可以使用通配符快捷方式。
任何以 结尾的位置都会将所有直接子项作为配置树导入。
与非通配符导入一样,每个配置树下的文件夹和文件的名称构成属性名称。configtree:
/*/
例如,给定以下卷:
etc/ config/ dbconfig/ db/ username password mqconfig/ mq/ username password
您可以用作导入位置:configtree:/etc/config/*/
spring.config.import=optional:configtree:/etc/config/*/
spring:
config:
import: "optional:configtree:/etc/config/*/"
这将添加 、 和 属性。db.username
db.password
mq.username
mq.password
使用通配符加载的目录按字母顺序排序。 如果您需要不同的顺序,则应将每个位置列为单独的导入 |
配置树还可用于 Docker 密钥。
当 Docker Swarm 服务被授予对 secret 的访问权限时,secret 将被挂载到容器中。
例如,如果名为 的密钥挂载在 location ,则可以使用以下内容对 Spring 环境可用:db.password
/run/secrets/
db.password
spring.config.import=optional:configtree:/run/secrets/
spring:
config:
import: "optional:configtree:/run/secrets/"
2.3.7. 属性占位符
和 中的值在使用时通过现有值进行筛选,因此您可以引用以前定义的值 (例如,从系统属性或环境变量)。
标准 property-placeholder 语法可以在值中的任何位置使用。
属性占位符还可以使用 a 将默认值与属性名称分开,例如 。application.properties
application.yaml
Environment
${name}
:
${name:default}
以下示例显示了带和不带默认值的占位符的用法:
app.name=MyApp
app.description=${app.name} is a Spring Boot application written by ${username:Unknown}
app:
name: "MyApp"
description: "${app.name} is a Spring Boot application written by ${username:Unknown}"
假设该属性尚未在其他位置设置,则其值为 .username
app.description
MyApp is a Spring Boot application written by Unknown
您应该始终使用其规范形式(kebab-case 仅使用小写字母)在占位符中引用属性名称。
这将允许 Spring Boot 使用与 relaxed binding 时相同的 logic 。 例如,将从文件以及系统环境中选取 和 form。
如果你改用,则不会被考虑。 |
您还可以使用此技术创建现有 Spring Boot 属性的“短”变体。 有关详细信息,请参阅 howto.html 操作方法。 |
2.3.8. 使用多文档文件
Spring Boot 允许您将单个物理文件拆分为多个 logical 文档,每个 logical 文档都是独立添加的。 文档按从上到下的顺序处理。 后面的文档可以覆盖前面文档中定义的属性。
对于文件,使用标准的 YAML 多文档语法。
三个连续的连字符表示一个文档的结尾和下一个文档的开头。application.yaml
例如,以下文件有两个逻辑文档:
spring:
application:
name: "MyApp"
---
spring:
application:
name: "MyCloudApp"
config:
activate:
on-cloud-platform: "kubernetes"
对于文件,使用 special 或 comment 来标记文档拆分:application.properties
#---
!---
spring.application.name=MyApp
#---
spring.application.name=MyCloudApp
spring.config.activate.on-cloud-platform=kubernetes
属性文件分隔符不得有任何前导空格,并且必须恰好包含三个连字符。 紧接在分隔符之前和之后的行不能是相同的注释前缀。 |
多文档属性文件通常与激活属性(如 .
有关详细信息,请参阅下一节。spring.config.activate.on-profile |
无法使用 or 注释加载多文档属性文件。@PropertySource @TestPropertySource |
2.3.9. 激活属性
有时,仅在满足某些条件时激活一组给定的属性很有用。 例如,您可能具有仅在特定配置文件处于活动状态时才相关的属性。
您可以使用 有条件地激活属性文档。spring.config.activate.*
可以使用以下激活属性:
财产 | 注意 |
---|---|
|
必须匹配的配置文件表达式,文档才能处于活动状态。 |
|
必须检测到 才能使文档处于活动状态。 |
例如,以下指定第二个文档仅在 Kubernetes 上运行时处于活动状态,并且仅当 “prod” 或 “staging” 配置文件处于活动状态时:
myprop=always-set
#---
spring.config.activate.on-cloud-platform=kubernetes
spring.config.activate.on-profile=prod | staging
myotherprop=sometimes-set
myprop:
"always-set"
---
spring:
config:
activate:
on-cloud-platform: "kubernetes"
on-profile: "prod | staging"
myotherprop: "sometimes-set"
2.4. 加密属性
Spring Boot 不提供对加密属性值的任何内置支持,但是,它确实提供了修改 Spring 中包含的值所需的钩子点。
该界面允许您在应用程序启动之前进行操作。
有关详细信息,请参阅 howto.html。Environment
EnvironmentPostProcessor
Environment
如果你需要一种安全的方式来存储凭证和密码,Spring Cloud Vault 项目提供了在 HashiCorp Vault 中存储外部化配置的支持。
2.5. 使用 YAML
YAML 是 JSON 的超集,因此是指定分层配置数据的便捷格式。
只要你的 Classpath 上有 SnakeYAML 库,该类就会自动支持 YAML 作为属性的替代。SpringApplication
如果您使用 “Starters”,则 SnakeYAML 由 .spring-boot-starter |
2.5.1. 将 YAML 映射到属性
YAML 文档需要从其分层格式转换为可与 Spring 一起使用的平面结构。
例如,请考虑以下 YAML 文档:Environment
environments:
dev:
url: "https://dev.example.com"
name: "Developer Setup"
prod:
url: "https://another.example.com"
name: "My Cool App"
为了从 访问这些属性,它们将被拼合,如下所示:Environment
environments.dev.url=https://dev.example.com
environments.dev.name=Developer Setup
environments.prod.url=https://another.example.com
environments.prod.name=My Cool App
同样,YAML 列表也需要扁平化。
它们表示为带有取消引用器的属性键。
例如,请考虑以下 YAML:[index]
my:
servers:
- "dev.example.com"
- "another.example.com"
前面的示例将转换为以下属性:
my.servers[0]=dev.example.com
my.servers[1]=another.example.com
使用该表示法的属性可以使用 Spring Boot 的类绑定到 Java 或对象。
有关更多详细信息,请参阅下面的“类型安全配置属性”部分。[index] List Set Binder |
无法使用 or 注解加载 YAML 文件。
因此,如果需要以这种方式加载值,则需要使用属性文件。@PropertySource @TestPropertySource |
2.6. 配置随机值
这对于注入随机值(例如,注入 secret 或测试用例)很有用。
它可以生成整数、长整型、uuid 或字符串,如以下示例所示:RandomValuePropertySource
my.secret=${random.value}
my.number=${random.int}
my.bignumber=${random.long}
my.uuid=${random.uuid}
my.number-less-than-ten=${random.int(10)}
my.number-in-range=${random.int[1024,65536]}
my:
secret: "${random.value}"
number: "${random.int}"
bignumber: "${random.long}"
uuid: "${random.uuid}"
number-less-than-ten: "${random.int(10)}"
number-in-range: "${random.int[1024,65536]}"
语法是其中 s 是任何字符 和 整数。
如果提供,则为最小值和最大值(不包括)。random.int*
OPEN value (,max) CLOSE
OPEN,CLOSE
value,max
max
value
max
2.7. 配置系统环境属性
Spring Boot 支持为环境属性设置前缀。
如果系统环境由具有不同配置要求的多个 Spring Boot 应用程序共享,这将非常有用。
可以直接在 上设置系统环境属性的前缀。SpringApplication
例如,如果将前缀设置为 ,则诸如 这样的属性也将像在系统环境中一样进行解析。input
remote.timeout
input.remote.timeout
2.8. 类型安全的配置属性
使用 annotation 注入配置属性有时会很麻烦,尤其是在您使用多个属性或数据本质上是分层的时。
Spring Boot 提供了一种使用属性的替代方法,该方法允许强类型 bean 管理和验证应用程序的配置。@Value("${property}")
2.8.1. JavaBean 属性绑定
可以绑定声明标准 JavaBean 属性的 Bean,如下例所示:
@ConfigurationProperties("my.service")
public class MyProperties {
private boolean enabled;
private InetAddress remoteAddress;
private final Security security = new Security();
public static class Security {
private String username;
private String password;
private List<String> roles = new ArrayList<>(Collections.singleton("USER"));
}
}
@ConfigurationProperties("my.service")
class MyProperties {
var isEnabled = false
var remoteAddress: InetAddress? = null
val security = Security()
class Security {
var username: String? = null
var password: String? = null
var roles: List<String> = ArrayList(setOf("USER"))
}
}
前面的 POJO 定义了以下属性:
-
my.service.enabled
,默认值为false
-
my.service.remote-address
,其类型可以从 强制转换。String
-
my.service.security.username
,其中包含一个嵌套的 “security” 对象,其名称由属性名称确定。 特别是,该类型根本没有在那里使用,并且可能是 。SecurityProperties
-
my.service.security.password
. -
my.service.security.roles
,其中的集合默认为 .String
USER
映射到 Spring Boot 中可用的类的属性(通过属性文件、YAML 文件、环境变量和其他机制进行配置)是公共 API,但类本身的访问器(getter/setter)不能直接使用。@ConfigurationProperties |
这种安排依赖于默认的空构造函数,并且 getter 和 setter 通常是强制性的,因为绑定是通过标准 Java Beans 属性描述符进行的,就像在 Spring MVC 中一样。 在以下情况下,可以省略 setter:
有些人使用 Project Lombok 自动添加 getter 和 setter。 确保 Lombok 不会为此类类型生成任何特定的构造函数,因为容器会自动使用它来实例化对象。 最后,仅考虑标准 Java Bean 属性,并且不支持对静态属性进行绑定。 |
2.8.2. 构造函数绑定
上一节中的示例可以以不可变的方式重写,如以下示例所示:
@ConfigurationProperties("my.service")
public class MyProperties {
public MyProperties(boolean enabled, InetAddress remoteAddress, Security security) {
this.enabled = enabled;
this.remoteAddress = remoteAddress;
this.security = security;
}
public static class Security {
public Security(String username, String password, @DefaultValue("USER") List<String> roles) {
this.username = username;
this.password = password;
this.roles = roles;
}
}
}
@ConfigurationProperties("my.service")
class MyProperties(val enabled: Boolean, val remoteAddress: InetAddress,
val security: Security) {
class Security(val username: String, val password: String,
@param:DefaultValue("USER") val roles: List<String>)
}
在此设置中,存在单个参数化构造函数意味着应使用构造函数绑定。
这意味着 Binder 将找到一个构造函数,其中包含您希望绑定的参数。
如果您的类具有多个构造函数,则注释可用于指定要用于构造函数绑定的构造函数。
要选择退出具有单个参数化构造函数的类的构造函数绑定,必须为构造函数添加注释或 made .
构造函数绑定可以与记录一起使用。
除非您的记录具有多个构造函数,否则无需使用 .@ConstructorBinding
@Autowired
private
@ConstructorBinding
构造函数绑定类的嵌套成员(如上例所示)也将通过其构造函数进行绑定。Security
可以使用 on constructor parameters 和 record 组件指定默认值。
将应用转换服务以将 Comments 的值强制转换为缺失属性的目标类型。@DefaultValue
String
参考前面的示例,如果没有属性绑定到 ,则实例将包含 的值 。
要使其包含非 null 实例,即使没有绑定到任何属性(使用 Kotlin 时,这将需要将 和 参数声明为可为 null,因为它们没有默认值),请使用空注解:Security
MyProperties
null
security
Security
username
password
Security
@DefaultValue
public MyProperties(boolean enabled, InetAddress remoteAddress, @DefaultValue Security security) {
this.enabled = enabled;
this.remoteAddress = remoteAddress;
this.security = security;
}
class MyProperties(val enabled: Boolean, val remoteAddress: InetAddress,
@DefaultValue val security: Security) {
class Security(val username: String?, val password: String?,
@param:DefaultValue("USER") val roles: List<String>)
}
要使用构造函数绑定,必须使用 or configuration 属性扫描启用该类。
您不能将构造函数绑定与由常规 Spring 机制创建的 bean(例如 bean、使用方法创建的 bean 或使用 bean 加载的 bean )一起使用@EnableConfigurationProperties @Component @Bean @Import )
|
要使用构造函数绑定,必须使用 .
如果您使用 Spring Boot 的 Gradle 插件,或者如果您使用 Maven 和 .-parameters spring-boot-starter-parent |
不建议使用 with,因为它主要用作返回类型。
因此,它不太适合 configuration property injection。
为了与其他类型的属性保持一致,如果声明了一个属性并且它没有值,则将绑定一个空值,而不是空值。java.util.Optional @ConfigurationProperties Optional null Optional |
要在属性名称中使用 reserved 关键字(如 ),请在 constructor 参数上使用注释。my.service.import @Name |
2.8.3. 启用 @ConfigurationProperties 注解类型
Spring Boot 提供了绑定类型并将其注册为 bean 的基础设施。
您可以逐个类启用配置属性,也可以启用配置属性扫描,其工作方式与组件扫描类似。@ConfigurationProperties
有时,带有 Comments 的类可能不适合扫描,例如,如果您正在开发自己的自动配置或想要有条件地启用它们。
在这些情况下,请使用注释指定要处理的类型列表。
这可以在任何类上完成,如以下示例所示:@ConfigurationProperties
@EnableConfigurationProperties
@Configuration
@Configuration(proxyBeanMethods = false)
@EnableConfigurationProperties(SomeProperties.class)
public class MyConfiguration {
}
@Configuration(proxyBeanMethods = false)
@EnableConfigurationProperties(SomeProperties::class)
class MyConfiguration
@ConfigurationProperties("some.properties")
public class SomeProperties {
}
@ConfigurationProperties("some.properties")
class SomeProperties
要使用配置属性扫描,请将注释添加到您的应用程序。
通常,它被添加到带有 Comments 的主应用程序类中,但可以添加到任何类中。
默认情况下,将从声明 Comments 的类的包中进行扫描。
如果要定义要扫描的特定包,可以按以下示例所示执行此操作:@ConfigurationPropertiesScan
@SpringBootApplication
@Configuration
@SpringBootApplication
@ConfigurationPropertiesScan({ "com.example.app", "com.example.another" })
public class MyApplication {
}
@SpringBootApplication
@ConfigurationPropertiesScan("com.example.app", "com.example.another")
class MyApplication
当使用配置属性 scanning 或通过 注册 bean 时,该 bean 具有约定的名称: ,其中 是注释中指定的环境键前缀,是 bean 的完全限定名称。
如果 Comments 未提供任何前缀,则仅使用 Bean 的完全限定名称。 假设它在包中,则上面示例的 bean 名称为 。 |
我们建议只处理环境,特别是不要从上下文中注入其他 bean。
对于极端情况,可以使用 setter 注入或框架提供的任何接口(例如,如果您需要访问 )。
如果您仍然希望使用构造函数注入其他 bean,则必须使用基于 JavaBean 的属性绑定对配置属性 bean 进行注释。@ConfigurationProperties
*Aware
EnvironmentAware
Environment
@Component
2.8.4. 使用 @ConfigurationProperties 注解类型
这种配置样式特别适用于外部 YAML 配置,如以下示例所示:SpringApplication
my:
service:
remote-address: 192.168.1.1
security:
username: "admin"
roles:
- "USER"
- "ADMIN"
要使用 bean,可以像注入任何其他 bean 一样注入它们,如以下示例所示:@ConfigurationProperties
@Service
public class MyService {
private final MyProperties properties;
public MyService(MyProperties properties) {
this.properties = properties;
}
public void openConnection() {
Server server = new Server(this.properties.getRemoteAddress());
server.start();
// ...
}
// ...
}
@Service
class MyService(val properties: MyProperties) {
fun openConnection() {
val server = Server(properties.remoteAddress)
server.start()
// ...
}
// ...
}
使用还可以生成元数据文件,IDE 可以使用这些文件为您自己的键提供自动完成功能。
有关详细信息,请参阅附录。@ConfigurationProperties |
2.8.5. 第三方配置
除了用于注释类外,您还可以在公共方法上使用它。
当您希望将属性绑定到不受控制的第三方组件时,这样做可能特别有用。@ConfigurationProperties
@Bean
要从属性中配置 Bean,请添加到其 Bean 注册中,如以下示例所示:Environment
@ConfigurationProperties
@Configuration(proxyBeanMethods = false)
public class ThirdPartyConfiguration {
@Bean
@ConfigurationProperties(prefix = "another")
public AnotherComponent anotherComponent() {
return new AnotherComponent();
}
}
@Configuration(proxyBeanMethods = false)
class ThirdPartyConfiguration {
@Bean
@ConfigurationProperties(prefix = "another")
fun anotherComponent(): AnotherComponent = AnotherComponent()
}
使用前缀定义的任何 JavaBean 属性都以类似于前面示例的方式映射到该 Bean 上。another
AnotherComponent
SomeProperties
2.8.6. 松散绑定
Spring Boot 使用一些宽松的规则将属性绑定到 bean,因此属性名称和 bean 属性名称之间不需要完全匹配。
这有用的常见示例包括以短划线分隔的环境属性(例如,binds to )和大写的环境属性(例如,binds to )。Environment
@ConfigurationProperties
Environment
context-path
contextPath
PORT
port
例如,请考虑以下类:@ConfigurationProperties
@ConfigurationProperties(prefix = "my.main-project.person")
public class MyPersonProperties {
private String firstName;
public String getFirstName() {
return this.firstName;
}
public void setFirstName(String firstName) {
this.firstName = firstName;
}
}
@ConfigurationProperties(prefix = "my.main-project.person")
class MyPersonProperties {
var firstName: String? = null
}
使用上述代码,可以使用以下属性名称:
财产 | 注意 |
---|---|
|
Kebab 大小写,建议在 和 YAML 文件中使用。 |
|
标准 Camel 大小写语法。 |
|
下划线表示法,这是在 和 YAML 文件中使用的替代格式。 |
|
大写格式,使用系统环境变量时推荐使用。 |
注释的值必须采用 kebab 大小写(小写并用 分隔,例如 )。prefix - my.main-project.person |
Property Source | 简单 | 列表 |
---|---|---|
属性文件 |
驼峰式大小写、kebab 大小写或下划线表示法 |
使用 或 逗号分隔值的标准列表语法 |
YAML 文件 |
驼峰式大小写、kebab 大小写或下划线表示法 |
标准 YAML 列表语法或逗号分隔值 |
环境变量 |
大写格式,下划线作为分隔符(请参阅从环境变量绑定)。 |
用下划线括起来的数值(请参见从环境变量绑定) |
系统属性 |
驼峰式大小写、kebab 大小写或下划线表示法 |
使用 或 逗号分隔值的标准列表语法 |
我们建议尽可能以小写 kebab 格式存储属性,例如 .my.person.first-name=Rod |
绑定映射
绑定到属性时,可能需要使用特殊的方括号表示法,以便保留原始值。
如果键未被 括起来,则任何非字母数字字符或已删除的字符。Map
key
[]
-
.
例如,考虑将以下属性绑定到 :Map<String,String>
my.map.[/key1]=value1
my.map.[/key2]=value2
my.map./key3=value3
my:
map:
"[/key1]": "value1"
"[/key2]": "value2"
"/key3": "value3"
对于 YAML 文件,括号需要用引号括起来,以便正确解析键。 |
上述属性将绑定到 with 和 作为映射中的键。
斜杠已被删除,因为它没有被方括号括起来。Map
/key1
/key2
key3
key3
当绑定到标量值时,其中包含的键不需要用 .
标量值包括枚举和包中除 .
Binding to 将保留键中的 the 并返回一个条目为 .
对于任何其他类型,如果包含.
例如,绑定到将返回带有条目的 Map,而将返回带有条目的 Map。.
[]
java.lang
Object
a.b=c
Map<String, String>
.
{"a.b"="c"}
key
.
a.b=c
Map<String, Object>
{"a"={"b"="c"}}
[a.b]=c
{"a.b"="c"}
从环境变量绑定
大多数操作系统对可用于环境变量的名称施加了严格的规则。
例如,Linux shell 变量只能包含字母 ( to 或 to )、数字 ( to ) 或下划线字符 ()。
按照惯例,Unix shell 变量的名称也将采用 UPPERCASE。a
z
A
Z
0
9
_
Spring Boot 的宽松绑定规则尽可能地设计为与这些命名限制兼容。
要将 canonical-form 中的属性名称转换为环境变量名称,您可以遵循以下规则:
-
将点 () 替换为下划线 ()。
.
_
-
删除所有破折号 ()。
-
-
转换为大写。
例如,configuration 属性将是名为 .spring.main.log-startup-info
SPRING_MAIN_LOGSTARTUPINFO
绑定到对象列表时,也可以使用环境变量。
要绑定到 ,元素编号应在变量名称中用下划线括起来。
例如,configuration 属性将使用名为 .List
my.service[0].other
MY_SERVICE_0_OTHER
对从环境变量进行绑定的支持将应用于属性源以及名称以 .systemEnvironment
-systemEnvironment
2.8.7. 合并复杂类型
在多个位置配置列表时,覆盖的工作原理是替换整个列表。
例如,假设对象具有 and 属性,默认情况下为 。
以下示例公开了 :MyPojo
name
description
null
MyPojo
MyProperties
@ConfigurationProperties("my")
public class MyProperties {
private final List<MyPojo> list = new ArrayList<>();
public List<MyPojo> getList() {
return this.list;
}
}
@ConfigurationProperties("my")
class MyProperties {
val list: List<MyPojo> = ArrayList()
}
请考虑以下配置:
my.list[0].name=my name
my.list[0].description=my description
#---
spring.config.activate.on-profile=dev
my.list[0].name=my another name
my:
list:
- name: "my name"
description: "my description"
---
spring:
config:
activate:
on-profile: "dev"
my:
list:
- name: "my another name"
如果配置文件未处于活动状态,则包含一个条目,如前所述。
但是,如果启用了配置文件,则 仍只包含一个条目(名称 和 描述 )。
此配置不会向列表添加第二个实例,也不会合并项目。dev
MyProperties.list
MyPojo
dev
list
my another name
null
MyPojo
在多个配置文件中指定 a 时,将使用优先级最高的配置文件(并且仅使用该配置文件)。
请考虑以下示例:List
my.list[0].name=my name
my.list[0].description=my description
my.list[1].name=another name
my.list[1].description=another description
#---
spring.config.activate.on-profile=dev
my.list[0].name=my another name
my:
list:
- name: "my name"
description: "my description"
- name: "another name"
description: "another description"
---
spring:
config:
activate:
on-profile: "dev"
my:
list:
- name: "my another name"
在前面的示例中,如果配置文件处于活动状态,则包含一个条目(名称为 ,描述为 )。
对于 YAML,逗号分隔列表和 YAML 列表都可用于完全覆盖列表的内容。dev
MyProperties.list
MyPojo
my another name
null
对于属性,您可以绑定从多个源提取的属性值。
但是,对于多个来源中的同一属性,将使用优先级最高的属性。
以下示例公开了 from :Map
Map<String, MyPojo>
MyProperties
@ConfigurationProperties("my")
public class MyProperties {
private final Map<String, MyPojo> map = new LinkedHashMap<>();
public Map<String, MyPojo> getMap() {
return this.map;
}
}
@ConfigurationProperties("my")
class MyProperties {
val map: Map<String, MyPojo> = LinkedHashMap()
}
请考虑以下配置:
my.map.key1.name=my name 1
my.map.key1.description=my description 1
#---
spring.config.activate.on-profile=dev
my.map.key1.name=dev name 1
my.map.key2.name=dev name 2
my.map.key2.description=dev description 2
my:
map:
key1:
name: "my name 1"
description: "my description 1"
---
spring:
config:
activate:
on-profile: "dev"
my:
map:
key1:
name: "dev name 1"
key2:
name: "dev name 2"
description: "dev description 2"
如果配置文件未处于活动状态,则包含一个具有键的条目(名称为 ,描述为 )。
但是,如果启用配置文件,则包含两个条目,键(名称为,描述为 )和(名称为 ,描述为 )。dev
MyProperties.map
key1
my name 1
my description 1
dev
map
key1
dev name 1
my description 1
key2
dev name 2
dev description 2
上述合并规则适用于所有属性源中的属性,而不仅仅是文件。 |
2.8.8. 属性转换
Spring Boot 在绑定到 bean 时尝试将外部应用程序属性强制为正确的类型。
如果需要自定义类型转换,则可以提供 bean(带有名为 )或自定义属性编辑器(通过 bean)或自定义(bean定义注释为 )。@ConfigurationProperties
ConversionService
conversionService
CustomEditorConfigurer
Converters
@ConfigurationPropertiesBinding
由于此 bean 是在应用程序生命周期的早期请求,因此请确保限制您正在使用的依赖项。
通常,您需要的任何依赖项在创建时可能未完全初始化。
如果配置键强制不需要,并且仅依赖于符合 的自定义转换器,则可能需要重命名自定义。ConversionService ConversionService @ConfigurationPropertiesBinding |
转换持续时间
Spring Boot 专门支持表示持续时间。
如果公开属性,则应用程序属性中的以下格式可用:java.time.Duration
-
常规表示形式(除非指定了 a,否则使用毫秒作为默认单位)
long
@DurationUnit
-
java.time.Duration
使用的标准 ISO-8601 格式 -
值和单位耦合的更易读的格式(表示 10 秒)
10s
请考虑以下示例:
@ConfigurationProperties("my")
public class MyProperties {
@DurationUnit(ChronoUnit.SECONDS)
private Duration sessionTimeout = Duration.ofSeconds(30);
private Duration readTimeout = Duration.ofMillis(1000);
}
@ConfigurationProperties("my")
class MyProperties {
@DurationUnit(ChronoUnit.SECONDS)
var sessionTimeout = Duration.ofSeconds(30)
var readTimeout = Duration.ofMillis(1000)
}
要指定 30 秒的会话超时,则 和 都是等效的。
可以采用以下任何形式指定 500 毫秒的读取超时:和 。30
PT30S
30s
500
PT0.5S
500ms
您还可以使用任何受支持的单位。 这些是:
-
ns
纳秒 -
us
微秒 -
ms
毫秒 -
s
几秒钟 -
m
几分钟 -
h
小时 -
d
几天
默认单位为毫秒,可以使用上面的示例所示进行覆盖。@DurationUnit
如果您更喜欢使用构造函数绑定,则可以公开相同的属性,如以下示例所示:
@ConfigurationProperties("my")
public class MyProperties {
public MyProperties(@DurationUnit(ChronoUnit.SECONDS) @DefaultValue("30s") Duration sessionTimeout,
@DefaultValue("1000ms") Duration readTimeout) {
this.sessionTimeout = sessionTimeout;
this.readTimeout = readTimeout;
}
}
@ConfigurationProperties("my")
class MyProperties(@param:DurationUnit(ChronoUnit.SECONDS) @param:DefaultValue("30s") val sessionTimeout: Duration,
@param:DefaultValue("1000ms") val readTimeout: Duration)
如果要升级属性,请确保定义单位(如果不是毫秒,则使用 )。
这样做提供了透明的升级路径,同时支持更丰富的格式。Long @DurationUnit |
转换期间
除了持续时间之外, Spring Boot 还可以使用类型。
应用程序属性中可以使用以下格式:java.time.Period
-
常规表示形式 (除非已指定 a,否则使用天数作为默认单位)
int
@PeriodUnit
-
java.time.Period
使用的标准 ISO-8601 格式 -
一种更简单的格式,其中值和单位对耦合(表示 1 年零 3 天)
1y3d
简单格式支持以下单位:
-
y
多年来 -
m
几个月 -
w
几周 -
d
几天
该类型从未实际存储周数,它是一个表示“7 天”的快捷方式。java.time.Period |
转换数据大小
Spring Framework 具有一个值类型,它以字节为单位表示大小。
如果公开属性,则应用程序属性中的以下格式可用:DataSize
DataSize
-
常规表示形式(除非指定了 a,否则使用 bytes 作为默认单位)
long
@DataSizeUnit
-
值和单位耦合的可读性更强的格式(表示 10 MB)
10MB
请考虑以下示例:
@ConfigurationProperties("my")
public class MyProperties {
@DataSizeUnit(DataUnit.MEGABYTES)
private DataSize bufferSize = DataSize.ofMegabytes(2);
private DataSize sizeThreshold = DataSize.ofBytes(512);
}
@ConfigurationProperties("my")
class MyProperties {
@DataSizeUnit(DataUnit.MEGABYTES)
var bufferSize = DataSize.ofMegabytes(2)
var sizeThreshold = DataSize.ofBytes(512)
}
指定缓冲区大小为 10 MB,并且等效。
可以将 256 字节的大小阈值指定为 或 。10
10MB
256
256B
您还可以使用任何受支持的单位。 这些是:
-
B
对于字节 -
KB
对于 KB -
MB
兆字节 -
GB
对于 GB -
TB
对于 TB
默认单位是 bytes,可以使用上面的示例所示进行覆盖。@DataSizeUnit
如果您更喜欢使用构造函数绑定,则可以公开相同的属性,如以下示例所示:
@ConfigurationProperties("my")
public class MyProperties {
public MyProperties(@DataSizeUnit(DataUnit.MEGABYTES) @DefaultValue("2MB") DataSize bufferSize,
@DefaultValue("512B") DataSize sizeThreshold) {
this.bufferSize = bufferSize;
this.sizeThreshold = sizeThreshold;
}
}
@ConfigurationProperties("my")
class MyProperties(@param:DataSizeUnit(DataUnit.MEGABYTES) @param:DefaultValue("2MB") val bufferSize: DataSize,
@param:DefaultValue("512B") val sizeThreshold: DataSize)
如果要升级属性,请确保定义单位(如果不是 bytes,则使用 )。
这样做提供了透明的升级路径,同时支持更丰富的格式。Long @DataSizeUnit |
2.8.9. @ConfigurationProperties 验证
Spring Boot 尝试在使用 Spring 的 Comments 对类进行 Comments 时对其进行验证。
您可以直接在 configuration 类上使用 JSR-303 约束 Comments。
为此,请确保 Classpath 上具有兼容的 JSR-303 实现,然后将约束 Comments 添加到字段中,如以下示例所示:@ConfigurationProperties
@Validated
jakarta.validation
@ConfigurationProperties("my.service")
@Validated
public class MyProperties {
@NotNull
private InetAddress remoteAddress;
}
@ConfigurationProperties("my.service")
@Validated
class MyProperties {
var remoteAddress: @NotNull InetAddress? = null
}
您还可以通过使用 注释创建配置属性的方法来触发验证。@Bean @Validated |
要确保始终为嵌套属性触发验证,即使未找到任何属性,也必须为关联的字段添加注释。
以下示例基于前面的示例:@Valid
MyProperties
@ConfigurationProperties("my.service")
@Validated
public class MyProperties {
@NotNull
private InetAddress remoteAddress;
@Valid
private final Security security = new Security();
public static class Security {
@NotEmpty
private String username;
}
}
@ConfigurationProperties("my.service")
@Validated
class MyProperties {
var remoteAddress: @NotNull InetAddress? = null
@Valid
val security = Security()
class Security {
@NotEmpty
var username: String? = null
}
}
您还可以通过创建名为 .
该方法应声明 。
配置属性验证器是在应用程序生命周期的早期创建的,将方法声明为 static 可以创建 bean,而不必实例化该类。
这样做可以避免早期实例化可能导致的任何问题。Validator
configurationPropertiesValidator
@Bean
static
@Bean
@Configuration
该模块包括一个公开所有 bean 的端点。
将 Web 浏览器指向或使用等效的 JMX 终端节点。
有关详细信息,请参阅“生产就绪功能”部分。spring-boot-actuator @ConfigurationProperties /actuator/configprops |
2.8.10. @ConfigurationProperties 与 @Value
注释是容器的核心功能,它不提供与类型安全配置属性相同的功能。
下表总结了 和 支持的功能:@Value
@ConfigurationProperties
@Value
特征 | @ConfigurationProperties |
@Value |
---|---|---|
是的 |
受限(请参阅下面的注释) |
|
是的 |
不 |
|
|
不 |
是的 |
如果您确实要使用 ,我们建议您使用属性名称的规范形式(kebab 大小写仅使用小写字母)来引用属性名称。
这将允许 Spring Boot 使用与 relaxed binding 时相同的 logic 。 例如,将从文件以及系统环境中选取 和 form。
如果你改用,则不会被考虑。 |
如果您为自己的组件定义了一组配置键,我们建议您将它们分组到一个带有 .
这样做将为您提供结构化的、类型安全的对象,您可以将其注入到自己的 bean 中。@ConfigurationProperties
SpEL
在解析 Application 属性文件和填充环境时,不会处理这些文件中的表达式。
但是,可以在 中编写表达式。
如果应用程序属性文件中的属性值是表达式,则在通过 .SpEL
@Value
SpEL
@Value
3. 配置文件
Spring 配置文件提供了一种分离应用程序配置的各个部分并使其仅在某些环境中可用的方法。
Any ,或者可以标记为 to limit 何时加载,如以下示例所示:@Component
@Configuration
@ConfigurationProperties
@Profile
@Configuration(proxyBeanMethods = false)
@Profile("production")
public class ProductionConfiguration {
// ...
}
@Configuration(proxyBeanMethods = false)
@Profile("production")
class ProductionConfiguration {
// ...
}
如果 bean 是通过而不是自动扫描注册的,则需要在具有 Comments 的类上指定 Comments。
在 被扫描的情况下,可以在类本身上指定。@ConfigurationProperties @EnableConfigurationProperties @Profile @Configuration @EnableConfigurationProperties @ConfigurationProperties @Profile @ConfigurationProperties |
您可以使用属性来指定哪些配置文件处于活动状态。
您可以使用本章前面介绍的任何方式指定属性。
例如,您可以将其包含在 中,如以下示例所示:spring.profiles.active
Environment
application.properties
spring.profiles.active=dev,hsqldb
spring:
profiles:
active: "dev,hsqldb"
您还可以使用以下开关在命令行中指定它:。--spring.profiles.active=dev,hsqldb
如果没有配置文件处于活动状态,则启用默认配置文件。
默认配置文件的名称是 ,并且可以使用 property 进行调整,如以下示例所示:default
spring.profiles.default
Environment
spring.profiles.default=none
spring:
profiles:
default: "none"
spring.profiles.active
,并且只能在非特定于配置文件的文档中使用。
这意味着它们不能包含在特定于配置文件的文件或 由 激活的文档中。spring.profiles.default
spring.config.activate.on-profile
例如,第二个文档配置无效:
# this document is valid
spring.profiles.active=prod
#---
# this document is invalid
spring.config.activate.on-profile=prod
spring.profiles.active=metrics
# this document is valid
spring:
profiles:
active: "prod"
---
# this document is invalid
spring:
config:
activate:
on-profile: "prod"
profiles:
active: "metrics"
3.1. 添加活动配置文件
该属性遵循与其他属性相同的排序规则:最高者优先。
这意味着您可以在 中指定活动配置文件,然后使用命令行开关替换它们。spring.profiles.active
PropertySource
application.properties
有时,将属性添加到活动配置文件中而不是替换它们非常有用。
该属性可用于在由该属性激活的用户档案之上添加活动用户档案。
入口点还具有用于设置其他配置文件的 Java API。
参见SpringApplication中的方法。spring.profiles.include
spring.profiles.active
SpringApplication
setAdditionalProfiles()
例如,当运行具有以下属性的应用程序时,即使它使用开关运行时,也会激活通用配置文件和本地配置文件:--spring.profiles.active
spring.profiles.include[0]=common
spring.profiles.include[1]=local
spring:
profiles:
include:
- "common"
- "local"
与 类似,只能在非配置文件特定的文档中使用。
这意味着它不能包含在由 激活的特定于配置文件的文件或文档中。spring.profiles.active spring.profiles.include spring.config.activate.on-profile |
如果给定配置文件处于活动状态,则下一节中介绍的配置文件组也可用于添加活动配置文件。
3.2. 配置文件组
有时,您在应用程序中定义和使用的配置文件过于精细,使用起来变得很麻烦。
例如,您可能拥有用于独立启用数据库和消息传递功能的 and 配置文件。proddb
prodmq
为了帮助解决这个问题, Spring Boot 允许你定义配置文件组。 配置文件组允许您为相关的配置文件组定义逻辑名称。
例如,我们可以创建一个由 our 和 profiles 组成的组。production
proddb
prodmq
spring.profiles.group.production[0]=proddb
spring.profiles.group.production[1]=prodmq
spring:
profiles:
group:
production:
- "proddb"
- "prodmq"
现在,我们的应用程序可以开始使用一次性激活 和 用户档案。--spring.profiles.active=production
production
proddb
prodmq
与 和 类似,只能在非配置文件特定的文档中使用。
这意味着它不能包含在由 激活的特定于配置文件的文件或文档中。spring.profiles.active spring.profiles.include spring.profiles.group spring.config.activate.on-profile |
3.3. 以编程方式设置 Profile
您可以通过在应用程序运行之前调用 来以编程方式设置活动配置文件。
也可以使用 Spring 的界面激活配置文件。SpringApplication.setAdditionalProfiles(…)
ConfigurableEnvironment
3.4. 特定于 Profile 的配置文件
通过 (或 ) 引用的文件的特定于配置文件的变体被视为文件并加载。
有关详细信息,请参阅“分析特定文件”。application.properties
application.yaml
@ConfigurationProperties
4. 日志记录
Spring Boot 使用 Commons Logging 进行所有内部日志记录,但将底层日志实现保持开放状态。 为 Java Util Logging、Log4j2 和 Logback 提供了默认配置。 在每种情况下,记录器都预先配置为使用控制台输出,并提供可选的文件输出。
默认情况下,如果使用“Starters”,则使用 Logback 进行日志记录。 还包括适当的 Logback 路由,以确保使用 Java Util Logging、Commons Logging、Log4J 或 SLF4J 的依赖库都能正常工作。
有很多可用于 Java 的日志记录框架。 如果上面的列表看起来令人困惑,请不要担心。 通常,您不需要更改日志记录依赖项,并且 Spring Boot 默认值可以正常工作。 |
将应用程序部署到 servlet 容器或应用程序服务器时,使用 Java Util Logging API 执行的日志记录不会路由到应用程序的日志中。 这可以防止容器或已部署到容器的其他应用程序执行的日志记录显示在应用程序的日志中。 |
4.1. 日志格式
Spring Boot 的默认日志输出类似于以下示例:
2024-10-04T13:32:30.226Z INFO 83455 --- [myapp] [ main] o.s.b.d.f.logexample.MyApplication : Starting MyApplication using Java 17.0.12 with PID 83455 (/opt/apps/myapp.jar started by myuser in /opt/apps/) 2024-10-04T13:32:30.232Z INFO 83455 --- [myapp] [ main] o.s.b.d.f.logexample.MyApplication : No active profile set, falling back to 1 default profile: "default" 2024-10-04T13:32:31.985Z INFO 83455 --- [myapp] [ main] o.s.b.w.embedded.tomcat.TomcatWebServer : Tomcat initialized with port 8080 (http) 2024-10-04T13:32:32.007Z INFO 83455 --- [myapp] [ main] o.apache.catalina.core.StandardService : Starting service [Tomcat] 2024-10-04T13:32:32.007Z INFO 83455 --- [myapp] [ main] o.apache.catalina.core.StandardEngine : Starting Servlet engine: [Apache Tomcat/10.1.30] 2024-10-04T13:32:32.086Z INFO 83455 --- [myapp] [ main] o.a.c.c.C.[Tomcat].[localhost].[/] : Initializing Spring embedded WebApplicationContext 2024-10-04T13:32:32.088Z INFO 83455 --- [myapp] [ main] w.s.c.ServletWebServerApplicationContext : Root WebApplicationContext: initialization completed in 1720 ms 2024-10-04T13:32:32.902Z INFO 83455 --- [myapp] [ main] o.s.b.w.embedded.tomcat.TomcatWebServer : Tomcat started on port 8080 (http) with context path '' 2024-10-04T13:32:32.920Z INFO 83455 --- [myapp] [ main] o.s.b.d.f.logexample.MyApplication : Started MyApplication in 3.859 seconds (process running for 4.634)
输出以下项目:
-
日期和时间:毫秒级精度,易于排序。
-
日志级别:、 、 、 或 。
ERROR
WARN
INFO
DEBUG
TRACE
-
进程 ID。
-
用于区分实际日志消息的开头的分隔符。
---
-
应用程序名称:括在方括号中(默认情况下,仅在设置时记录)
spring.application.name
-
线程名称:括在方括号中(对于控制台输出,可能会被截断)。
-
Correlation ID:如果启用了跟踪(在上面的示例中未显示)
-
Logger name:这通常是源类名称(通常缩写)。
-
日志消息。
Logback 没有级别。
它映射到 。FATAL ERROR |
如果您有属性但不希望它被记录下来,则可以设置为 .spring.application.name logging.include-application-name false |
4.2. 控制台输出
默认日志配置在写入消息时将消息回显到控制台。
默认情况下,将记录 -level、-level 和 -level 消息。
您还可以通过使用标志启动应用程序来启用 “debug” 模式。ERROR
WARN
INFO
--debug
$ java -jar myapp.jar --debug
您还可以在 .debug=true application.properties |
启用调试模式后,将配置一系列核心 Logger(嵌入式容器、Hibernate 和 Spring Boot)以输出更多信息。
启用 debug 模式不会将应用程序配置为记录所有具有 level 的消息。DEBUG
或者,您可以通过使用标志启动应用程序(或在您的 .
这样做可以为选定的核心 Logger(嵌入式容器、Hibernate 模式生成和整个 Spring 产品组合)启用跟踪日志记录。--trace
trace=true
application.properties
4.2.1. 颜色编码输出
如果您的终端支持 ANSI,则使用颜色输出来提高可读性。
您可以设置为支持的值以覆盖自动检测。spring.output.ansi.enabled
颜色编码是使用转换词配置的。
在最简单的形式中,转换器根据对数级别对输出进行着色,如以下示例所示:%clr
%clr(%5p)
下表描述了对数级别到颜色的映射:
水平 | 颜色 |
---|---|
|
红 |
|
红 |
|
黄色 |
|
绿 |
|
绿 |
|
绿 |
或者,您可以通过将颜色或样式作为转换选项来指定应使用的颜色或样式。 例如,要使文本变为黄色,请使用以下设置:
%clr(%d{yyyy-MM-dd'T'HH:mm:ss.SSSXXX}){yellow}
支持以下颜色和样式:
-
blue
-
cyan
-
faint
-
green
-
magenta
-
red
-
yellow
4.3. 文件输出
默认情况下, Spring Boot 仅记录到控制台,不写入日志文件。
如果除了控制台输出之外,还想写入日志文件,则需要设置 or 属性(例如,在您的 中)。
如果两个属性都设置了,则忽略并仅使用。logging.file.name
logging.file.path
application.properties
logging.file.path
logging.file.name
下表显示了如何结合使用这些属性:logging.*
logging.file.name |
logging.file.path |
描述 |
---|---|---|
(无) |
(无) |
仅限控制台日志记录。 |
特定文件(例如 |
(无) |
写入 指定的位置。
该位置可以是绝对位置,也可以是相对于当前目录的位置。 |
(无) |
特定目录(例如 |
写入 指定的目录。
该目录可以是绝对目录,也可以是相对于当前目录的目录。 |
特定文件 |
特定目录 |
写入 并忽略 指定的位置。
该位置可以是绝对位置,也可以是相对于当前目录的位置。 |
日志文件在达到 10 MB 时轮换,与控制台输出一样,默认情况下会记录 -level、-level 和 -level 消息。ERROR
WARN
INFO
日志记录属性独立于实际的日志记录基础结构。
因此,特定的配置键(例如 Logback)不受 Spring Boot 的管理。logback.configurationFile |
4.4. 文件旋转
如果您使用的是 Logback,则可以使用 or 文件微调日志轮换设置。
对于所有其他日志记录系统,您需要直接自己配置轮换设置(例如,如果您使用 Log4j2,则可以添加 or 文件)。application.properties
application.yaml
log4j2.xml
log4j2-spring.xml
支持以下轮换策略属性:
名字 | 描述 |
---|---|
|
用于创建日志存档的文件名模式。 |
|
是否应在应用程序启动时进行日志存档清理。 |
|
日志文件存档前的最大大小。 |
|
日志存档在被删除之前可以占用的最大大小。 |
|
要保留的存档日志文件的最大数量(默认为 7)。 |
4.5. 日志级别
所有受支持的日志记录系统都可以在 Spring 中设置 Logger 级别(例如,in ),方法是使用 where 是 TRACE、DEBUG、INFO、WARN、ERROR、FATAL 或 OFF 之一。
可以使用 来配置 Logger。Environment
application.properties
logging.level.<logger-name>=<level>
level
root
logging.level.root
以下示例显示了 中的潜在日志记录设置:application.properties
logging.level.root=warn
logging.level.org.springframework.web=debug
logging.level.org.hibernate=error
logging:
level:
root: "warn"
org.springframework.web: "debug"
org.hibernate: "error"
还可以使用环境变量设置日志记录级别。
例如,将设置为 .LOGGING_LEVEL_ORG_SPRINGFRAMEWORK_WEB=DEBUG
org.springframework.web
DEBUG
上述方法仅适用于包级日志记录。
由于松散绑定始终将环境变量转换为小写,因此无法以这种方式为单个类配置日志记录。
如果需要为类配置日志记录,可以使用 SPRING_APPLICATION_JSON 变量。 |
4.6. 日志组
能够将相关的 logger 分组在一起,以便可以同时配置它们通常很有用。 例如,您可能通常会更改所有与 Tomcat 相关的 Logger 的日志记录级别,但您无法轻松记住顶级包。
为了帮助解决这个问题, Spring Boot 允许您在 Spring 中定义 logging groups 。
例如,以下是如何通过将 “tomcat” 组添加到您的 :Environment
application.properties
logging.group.tomcat=org.apache.catalina,org.apache.coyote,org.apache.tomcat
logging:
group:
tomcat: "org.apache.catalina,org.apache.coyote,org.apache.tomcat"
定义后,您可以使用一行更改组中所有 logger 的级别:
logging.level.tomcat=trace
logging:
level:
tomcat: "trace"
Spring Boot 包括以下预定义的日志记录组,这些组可以开箱即用:
名字 | Logging |
---|---|
Web |
|
SQL |
|
4.7. 使用日志关闭钩子
为了在应用程序终止时释放日志记录资源,提供了一个关闭钩子,该钩子将在 JVM 退出时触发日志系统清理。
除非您的应用程序部署为 war 文件,否则会自动注册此关闭钩子。
如果您的应用程序具有复杂的上下文层次结构,则 shutdown 钩子可能无法满足您的需求。
如果没有,请禁用 shutdown 钩子并调查底层日志记录系统直接提供的选项。
例如,Logback 提供了上下文选择器,允许在自己的上下文中创建每个 Logger。
你可以使用该属性来禁用 shutdown 钩子。
将其设置为 将禁用注册。
您可以在 or 文件中设置该属性:logging.register-shutdown-hook
false
application.properties
application.yaml
logging.register-shutdown-hook=false
logging:
register-shutdown-hook: false
4.8. 自定义日志配置
可以通过在 Classpath 中包含适当的库来激活各种日志记录系统,并且可以通过在 Classpath 的根目录中或以下 Spring 属性指定的位置提供合适的配置文件来进一步自定义:。Environment
logging.config
你可以通过使用system属性来强制 Spring Boot 使用特定的日志记录系统。
该值应为 implementation 的完全限定类名。
您还可以使用值 .org.springframework.boot.logging.LoggingSystem
LoggingSystem
none
由于日志记录是在创建之前初始化的,因此无法从 Spring 文件中控制日志记录。
更改日志记录系统或完全禁用它的唯一方法是通过 System properties。ApplicationContext @PropertySources @Configuration |
根据您的日志记录系统,将加载以下文件:
测井系统 | 定制 |
---|---|
Logback (日志返回) |
|
日志 4j2 |
|
JDK(Java Util 日志记录) |
|
如果可能,我们建议您对日志记录配置使用 variants(例如,而不是 )。
如果使用标准配置位置, Spring 则无法完全控制日志初始化。-spring logback-spring.xml logback.xml |
Java Util Logging 存在已知的类加载问题,这些问题会导致在从“可执行 jar”运行时出现问题。 我们建议您尽可能避免从“可执行 jar”运行时使用它。 |
为了帮助进行自定义,将一些其他属性从 Spring 属性传输到 System 属性。
这允许记录系统配置来使用属性。例如,在环境变量中设置或设置为环境变量将导致设置 System 属性。
下表描述了传输的属性:Environment
logging.file.name
application.properties
LOGGING_FILE_NAME
LOG_FILE
Spring 环境 | 系统属性 | 评论 |
---|---|---|
|
|
记录异常时使用的转换字。 |
|
|
如果已定义,则在默认日志配置中使用它。 |
|
|
如果已定义,则在默认日志配置中使用它。 |
|
|
要在控制台上使用的日志模式 (stdout)。 |
|
|
日志日期格式的 Appender 模式。 |
|
|
用于控制台日志记录的字符集。 |
|
|
用于控制台日志记录的日志级别阈值。 |
|
|
要在文件中使用的日志模式(如果已启用)。 |
|
|
用于文件日志记录的字符集(如果已启用)。 |
|
|
用于文件日志记录的日志级别阈值。 |
|
|
渲染日志级别时使用的格式 (default )。 |
|
|
当前进程 ID(如果可能且尚未定义为 OS 环境变量,则已发现)。 |
如果使用 Logback,则还会传输以下属性:
Spring 环境 | 系统属性 | 评论 |
---|---|---|
|
|
滚动日志文件名的模式(默认)。 |
|
|
是否在启动时清理存档日志文件。 |
|
|
最大日志文件大小。 |
|
|
要保留的日志备份的总大小。 |
|
|
要保留的存档日志文件的最大数量。 |
所有受支持的日志记录系统都可以在解析其配置文件时查阅 System properties。
有关示例,请参阅中的默认配置:spring-boot.jar
如果要在日志记录属性中使用占位符,则应使用 Spring Boot 的语法,而不是底层框架的语法。
值得注意的是,如果使用 Logback,则应用作属性名称与其默认值之间的分隔符,而不是使用 。 |
您可以通过仅覆盖(或使用 Logback)将 MDC 和其他临时内容添加到日志行中。
例如,如果使用 ,则默认日志格式包含 “user” 的 MDC 条目(如果存在),如以下示例所示。 2019-08-30 12:30:04.031 user:someone INFO 22174 --- [ nio-8080-exec-0] demo.Controller Handling authenticated request |
4.9. Logback 扩展
Spring Boot 包含许多 Logback 扩展,可以帮助进行高级配置。
您可以在配置文件中使用这些扩展。logback-spring.xml
由于标准配置文件加载得太早,因此不能在其中使用扩展。
您需要使用或定义属性。logback.xml logback-spring.xml logging.config |
这些扩展不能与 Logback 的配置扫描一起使用。 如果尝试这样做,则对配置文件进行更改会导致记录类似于以下内容之一的错误: |
ERROR in ch.qos.logback.core.joran.spi.Interpreter@4:71 - no applicable action for [springProperty], current ElementPath is [[configuration][springProperty]] ERROR in ch.qos.logback.core.joran.spi.Interpreter@4:71 - no applicable action for [springProfile], current ElementPath is [[configuration][springProfile]]
4.9.1. 特定于 Profile 的配置
该标记允许您根据活动的 Spring 配置文件选择性地包含或排除配置部分。
元素中的任何位置都支持 Profile 部分。
使用 属性 指定接受配置的配置文件。
标记可以包含配置文件名称(例如 )或配置文件表达式。
配置文件表达式允许表示更复杂的配置文件逻辑,例如 .
有关更多详细信息,请查看 Spring Framework 参考指南。
以下清单显示了三个示例配置文件:<springProfile>
<configuration>
name
<springProfile>
staging
production & (eu-central | eu-west)
<springProfile name="staging">
<!-- configuration to be enabled when the "staging" profile is active -->
</springProfile>
<springProfile name="dev | staging">
<!-- configuration to be enabled when the "dev" or "staging" profiles are active -->
</springProfile>
<springProfile name="!production">
<!-- configuration to be enabled when the "production" profile is not active -->
</springProfile>
4.9.2. 环境属性
该标签允许您公开 Spring 中的属性,以便在 Logback 中使用。
如果要在 Logback 配置中访问文件中的值,则这样做可能很有用。
该标签的工作方式与 Logback 的标准标签类似。
但是,您不是指定直接 ,而是指定属性的 (来自 )。
如果需要将属性存储在范围以外的其他位置,则可以使用该属性。
如果需要回退值(如果未在 中设置属性),则可以使用该属性。
以下示例显示了如何公开要在 Logback 中使用的属性:<springProperty>
Environment
application.properties
<property>
value
source
Environment
local
scope
Environment
defaultValue
<springProperty scope="context" name="fluentHost" source="myapp.fluentd.host"
defaultValue="localhost"/>
<appender name="FLUENT" class="ch.qos.logback.more.appenders.DataFluentAppender">
<remoteHost>${fluentHost}</remoteHost>
...
</appender>
必须在 kebab 大小写中指定 (例如 )。
但是,可以使用宽松规则将属性添加到 中。source my.property-name Environment |
4.10. Log4j2 扩展
Spring Boot 包含许多对 Log4j2 的扩展,这些扩展可以帮助进行高级配置。
您可以在任何配置文件中使用这些扩展。log4j2-spring.xml
由于标准配置文件加载得太早,因此不能在其中使用扩展。
您需要使用或定义属性。log4j2.xml log4j2-spring.xml logging.config |
这些扩展取代了 Log4J 提供的 Spring Boot 支持。
您应该确保不要在构建中包含该模块。org.apache.logging.log4j:log4j-spring-boot |
4.10.1. 特定于 Profile 的配置
该标记允许您根据活动的 Spring 配置文件选择性地包含或排除配置部分。
元素中的任何位置都支持 Profile 部分。
使用 属性 指定接受配置的配置文件。
标记可以包含配置文件名称(例如 )或配置文件表达式。
配置文件表达式允许表示更复杂的配置文件逻辑,例如 .
有关更多详细信息,请查看 Spring Framework 参考指南。
以下清单显示了三个示例配置文件:<SpringProfile>
<Configuration>
name
<SpringProfile>
staging
production & (eu-central | eu-west)
<SpringProfile name="staging">
<!-- configuration to be enabled when the "staging" profile is active -->
</SpringProfile>
<SpringProfile name="dev | staging">
<!-- configuration to be enabled when the "dev" or "staging" profiles are active -->
</SpringProfile>
<SpringProfile name="!production">
<!-- configuration to be enabled when the "production" profile is not active -->
</SpringProfile>
4.10.2. 环境属性查找
如果要在 Log4j2 配置中引用 Spring 中的属性,可以使用带前缀的查找。
如果要在 Log4j2 配置中访问文件中的值,则这样做可能很有用。Environment
spring:
application.properties
以下示例显示如何设置从 Spring 读取的名为 Log4j2 的属性:applicationName
spring.application.name
Environment
<Properties>
<Property name="applicationName">${spring:spring.application.name}</Property>
</Properties>
查找键应以 kebab 大小写指定(例如 )。my.property-name |
4.10.3. Log4j2 系统属性
Log4j2 支持许多可用于配置各种项目的系统属性。
例如,system 属性可用于配置是否将尝试在 Windows 上使用 Jansi 输出流。log4j2.skipJansi
ConsoleAppender
Log4j2 初始化后加载的所有系统属性都可以从 Spring 获取。
例如,您可以添加到文件中以在 Windows 上使用 Jansi。Environment
log4j2.skipJansi=false
application.properties
ConsoleAppender
仅当系统属性和 OS 环境变量不包含正在加载的值时,才会考虑 Spring。Environment |
在早期 Log4j2 初始化期间加载的系统属性无法引用 Spring 。
例如,在 Spring Environment 可用之前,使用了 Log4j2 用于允许选择默认 Log4j2 实现的属性。Environment |
5. 国际化
Spring Boot 支持本地化消息,以便您的应用程序可以满足不同语言偏好的用户。
默认情况下, Spring Boot 在 Classpath 的根目录中查找是否存在资源包。messages
当已配置资源包的默认属性文件可用时(默认情况下),自动配置将适用。
如果您的资源包仅包含特定于语言的属性文件,则需要添加默认值。
如果未找到与任何已配置基本名称匹配的属性文件,则不会有 auto-configured 。messages.properties MessageSource |
可以使用命名空间配置资源包的基本名称以及其他几个属性,如以下示例所示:spring.messages
spring.messages.basename=messages,config.i18n.messages
spring.messages.fallback-to-system-locale=false
spring:
messages:
basename: "messages,config.i18n.messages"
fallback-to-system-locale: false
spring.messages.basename 支持逗号分隔的位置列表,可以是 package 限定符或从 Classpath 根解析的资源。 |
有关更多支持的选项,请参见 MessageSourceProperties
。
6. 面向方面的编程
Spring Boot 为面向方面的编程 (AOP) 提供自动配置。 您可以在 Spring Framework 参考文档中了解有关 AOP with Spring 的更多信息。
默认情况下,Spring Boot 的自动配置将 Spring AOP 配置为使用 CGLib 代理。
要改用 JDK 代理,请设置为 .spring.aop.proxy-target-class
false
如果 AspectJ 在 Classpath 上,则 Spring Boot 的自动配置将自动启用 AspectJ 自动代理,因此不是必需的。@EnableAspectJAutoProxy
7. JSON
Spring Boot 提供与三个 JSON 映射库的集成:
-
格森
-
Jackson
-
JSON-B 格式
Jackson 是首选和默认库。
7.1. Jackson
提供了 Jackson 的自动配置,并且 Jackson 是 的一部分。
当 Jackson 位于 Classpath 上时,将自动配置 bean。
提供了几个配置属性,用于自定义 ObjectMapper
的配置。spring-boot-starter-json
ObjectMapper
7.1.1. 自定义序列化器和反序列化器
如果您使用 Jackson 序列化和反序列化 JSON 数据,则可能需要编写自己的 and 类。
自定义序列化器通常通过模块向 Jackson 注册,但 Spring Boot 提供了一个替代 Comments,可以更轻松地直接注册 Spring Beans。JsonSerializer
JsonDeserializer
@JsonComponent
您可以直接在 或 implementations 上使用注释。
您还可以在包含序列化器/反序列化器作为内部类的类上使用它,如以下示例所示:@JsonComponent
JsonSerializer
JsonDeserializer
KeyDeserializer
@JsonComponent
public class MyJsonComponent {
public static class Serializer extends JsonSerializer<MyObject> {
@Override
public void serialize(MyObject value, JsonGenerator jgen, SerializerProvider serializers) throws IOException {
jgen.writeStartObject();
jgen.writeStringField("name", value.getName());
jgen.writeNumberField("age", value.getAge());
jgen.writeEndObject();
}
}
public static class Deserializer extends JsonDeserializer<MyObject> {
@Override
public MyObject deserialize(JsonParser jsonParser, DeserializationContext ctxt) throws IOException {
ObjectCodec codec = jsonParser.getCodec();
JsonNode tree = codec.readTree(jsonParser);
String name = tree.get("name").textValue();
int age = tree.get("age").intValue();
return new MyObject(name, age);
}
}
}
@JsonComponent
class MyJsonComponent {
class Serializer : JsonSerializer<MyObject>() {
@Throws(IOException::class)
override fun serialize(value: MyObject, jgen: JsonGenerator, serializers: SerializerProvider) {
jgen.writeStartObject()
jgen.writeStringField("name", value.name)
jgen.writeNumberField("age", value.age)
jgen.writeEndObject()
}
}
class Deserializer : JsonDeserializer<MyObject>() {
@Throws(IOException::class, JsonProcessingException::class)
override fun deserialize(jsonParser: JsonParser, ctxt: DeserializationContext): MyObject {
val codec = jsonParser.codec
val tree = codec.readTree<JsonNode>(jsonParser)
val name = tree["name"].textValue()
val age = tree["age"].intValue()
return MyObject(name, age)
}
}
}
中的所有 bean 都会自动注册到 Jackson。
因为 是元注释的,所以通常的组件扫描规则适用。@JsonComponent
ApplicationContext
@JsonComponent
@Component
Spring Boot 还提供了JsonObjectSerializer
和JsonObjectDeserializer
基类,它们在序列化对象时为标准 Jackson 版本提供了有用的替代方案。
有关详细信息,请参阅 Javadoc 中的 JsonObjectSerializer
和 JsonObjectDeserializer
。
上面的示例可以重写为使用 /,如下所示:JsonObjectSerializer
JsonObjectDeserializer
@JsonComponent
public class MyJsonComponent {
public static class Serializer extends JsonObjectSerializer<MyObject> {
@Override
protected void serializeObject(MyObject value, JsonGenerator jgen, SerializerProvider provider)
throws IOException {
jgen.writeStringField("name", value.getName());
jgen.writeNumberField("age", value.getAge());
}
}
public static class Deserializer extends JsonObjectDeserializer<MyObject> {
@Override
protected MyObject deserializeObject(JsonParser jsonParser, DeserializationContext context, ObjectCodec codec,
JsonNode tree) throws IOException {
String name = nullSafeValue(tree.get("name"), String.class);
int age = nullSafeValue(tree.get("age"), Integer.class);
return new MyObject(name, age);
}
}
}
`object`
@JsonComponent
class MyJsonComponent {
class Serializer : JsonObjectSerializer<MyObject>() {
@Throws(IOException::class)
override fun serializeObject(value: MyObject, jgen: JsonGenerator, provider: SerializerProvider) {
jgen.writeStringField("name", value.name)
jgen.writeNumberField("age", value.age)
}
}
class Deserializer : JsonObjectDeserializer<MyObject>() {
@Throws(IOException::class)
override fun deserializeObject(jsonParser: JsonParser, context: DeserializationContext,
codec: ObjectCodec, tree: JsonNode): MyObject {
val name = nullSafeValue(tree["name"], String::class.java)
val age = nullSafeValue(tree["age"], Int::class.java)
return MyObject(name, age)
}
}
}
8. 任务执行和调度
如果上下文中没有 bean,Spring Boot 会自动配置一个 .
启用虚拟线程(使用 Java 21+ 并设置为 )后,这将是使用虚拟线程。
否则,它将是一个具有合理默认值的 a。
在任何一种情况下,自动配置的执行程序都将自动用于:Executor
AsyncTaskExecutor
spring.threads.virtual.enabled
true
SimpleAsyncTaskExecutor
ThreadPoolTaskExecutor
-
异步任务执行 (
@EnableAsync
) -
Spring for GraphQL 对控制器方法返回值的异步处理
Callable
-
Spring MVC 的异步请求处理
-
Spring WebFlux 的阻塞执行支持
如果您在上下文中定义了自定义,则常规任务执行(即)和 Spring for GraphQL 都将使用它。
但是,Spring MVC 和 Spring WebFlux 支持仅在它是一个实现(名为 )时才会使用它。
根据您的目标排列方式,您可以将 your 更改为 an 或同时定义 an 和 an 包装 custom 。 自动配置允许您轻松创建实例,这些实例可以重现自动配置默认执行的操作。 |
当 a 被自动配置时,线程池使用 8 个核心线程,这些线程可以根据负载进行扩展和收缩。
可以使用命名空间对这些默认设置进行微调,如以下示例所示:ThreadPoolTaskExecutor
spring.task.execution
spring.task.execution.pool.max-size=16
spring.task.execution.pool.queue-capacity=100
spring.task.execution.pool.keep-alive=10s
spring:
task:
execution:
pool:
max-size: 16
queue-capacity: 100
keep-alive: "10s"
这会将线程池更改为使用有界队列,以便在队列已满(100 个任务)时,线程池增加到最多 16 个线程。 池的收缩更加激进,因为线程在空闲 10 秒(而不是默认 60 秒)时被回收。
如果需要将计划程序与计划任务执行相关联(例如使用),也可以自动配置计划程序。@EnableScheduling
如果启用了虚拟线程(使用 Java 21+ 并设置为 ),这将是使用虚拟线程。
这将忽略任何与池化相关的属性。spring.threads.virtual.enabled
true
SimpleAsyncTaskScheduler
SimpleAsyncTaskScheduler
如果未启用虚拟线程,它将是具有合理默认值的。
默认情况下,它使用一个线程,并且可以使用命名空间对其设置进行微调,如以下示例所示:ThreadPoolTaskScheduler
ThreadPoolTaskScheduler
spring.task.scheduling
spring.task.scheduling.thread-name-prefix=scheduling-
spring.task.scheduling.pool.size=2
spring:
task:
scheduling:
thread-name-prefix: "scheduling-"
pool:
size: 2
如果需要创建自定义执行程序或调度程序,则 bean、bean、bean 和 a 在上下文中可用。
如果启用了 和 bean(使用 Java 21+ 并设置为 ),则它们会自动配置为使用虚拟线程。ThreadPoolTaskExecutorBuilder
SimpleAsyncTaskExecutorBuilder
ThreadPoolTaskSchedulerBuilder
SimpleAsyncTaskSchedulerBuilder
SimpleAsyncTaskExecutorBuilder
SimpleAsyncTaskSchedulerBuilder
spring.threads.virtual.enabled
true
9. 测试
Spring Boot 提供了许多 Utilities 和 Comments,以帮助测试您的应用程序。
测试支持由两个模块提供:包含核心项,并支持测试的自动配置。spring-boot-test
spring-boot-test-autoconfigure
大多数开发人员使用“Starter”,它导入 Spring Boot 测试模块以及 JUnit Jupiter、AssertJ、Hamcrest 和许多其他有用的库。spring-boot-starter-test
如果您有使用 JUnit 4 的测试,则可以使用 JUnit 5 的 vintage 引擎来运行它们。
要使用 vintage 引擎,请添加对 的依赖项,如以下示例所示:
|
hamcrest-core
被排除在外,赞成 是 的一部分。org.hamcrest:hamcrest
spring-boot-starter-test
9.1. 测试范围依赖项
“Starter” (在 中) 包含以下提供的库:spring-boot-starter-test
test
scope
-
JUnit 5:Java 应用程序单元测试的事实标准。
-
Spring Test & Spring Boot Test:对Spring Boot应用程序的实用程序和集成测试支持。
-
AssertJ:一个 Fluent 断言库。
-
Hamcrest:匹配器对象库(也称为约束或谓词)。
-
Mockito:一个 Java 模拟框架。
-
JSONassert:JSON 的断言库。
-
JsonPath:JSON 的 XPath。
-
Awaitility:用于测试异步系统的库。
我们通常会发现这些公共库在编写测试时很有用。 如果这些库不符合您的需求,您可以添加自己的其他测试依赖项。
9.2. 测试 Spring 应用程序
依赖项注入的主要优点之一是它应该使您的代码更易于进行单元测试。
您可以使用 operator 实例化对象,甚至无需涉及 Spring。
您还可以使用 mock 对象而不是真正的依赖项。new
通常,您需要超越 unit testing 并开始集成测试(使用 Spring)。
能够在不需要部署应用程序或不需要连接到其他基础设施的情况下执行集成测试非常有用。ApplicationContext
Spring Framework 包括一个用于此类集成测试的专用测试模块。
您可以直接向 “Starter” 声明依赖项,也可以使用 “Starter” 以传递方式拉取它。org.springframework:spring-test
spring-boot-starter-test
如果您以前没有使用过该模块,则应首先阅读 Spring Framework 参考文档的相关部分。spring-test
9.3. 测试 Spring Boot 应用程序
Spring Boot 应用程序是 Spring ,因此除了通常使用普通 Spring 上下文所做的操作之外,无需执行任何特别操作来测试它。ApplicationContext
默认情况下,Spring Boot 的外部属性、日志记录和其他功能仅在用于创建它时才会安装在上下文中。SpringApplication |
Spring Boot 提供了一个 Comments,当您需要 Spring Boot 功能时,可以将其用作标准 Comments 的替代方法。
Comments 的工作原理是通过 SpringApplication
创建测试中使用的ApplicationContext
。
除了许多其他注释之外,还提供了用于测试应用程序的更具体切片的注释。@SpringBootTest
spring-test
@ContextConfiguration
@SpringBootTest
如果您使用的是 JUnit 4,请不要忘记也添加到您的测试中,否则注释将被忽略。
如果您使用的是 JUnit 5,则无需添加等效的 as,并且其他注释已经使用它进行了注释。@RunWith(SpringRunner.class) @ExtendWith(SpringExtension.class) @SpringBootTest @…Test |
默认情况下,不会启动服务器。
您可以使用 的 属性 of 进一步优化测试的运行方式:@SpringBootTest
webEnvironment
@SpringBootTest
-
MOCK
(默认) :加载 Web 并提供模拟 Web 环境。 使用此注释时,嵌入式服务器不会启动。 如果 Web 环境在 Classpath 上不可用,则此模式透明地回退到创建常规的 non-web 。 它可以与@AutoConfigureMockMvc
或@AutoConfigureWebTestClient
结合使用,以便对 Web 应用程序进行基于模拟的测试。ApplicationContext
ApplicationContext
-
RANDOM_PORT
:加载 并提供真实的 Web 环境。 嵌入式服务器启动并侦听随机端口。WebServerApplicationContext
-
DEFINED_PORT
:加载 并提供真实的 Web 环境。 嵌入式服务器启动并侦听定义的端口(来自 )或默认端口 。WebServerApplicationContext
application.properties
8080
-
NONE
:使用 但不提供任何 Web 环境(模拟或其他)加载。ApplicationContext
SpringApplication
如果您的测试是 ,则默认情况下,它会在每个测试方法结束时回滚事务。
但是,由于将这种安排与 OR 一起使用会隐式提供真实的 servlet 环境,因此 HTTP 客户端和服务器在单独的线程中运行,因此在单独的事务中运行。
在这种情况下,在服务器上启动的任何事务都不会回滚。@Transactional RANDOM_PORT DEFINED_PORT |
@SpringBootTest 如果您的应用程序对 Management Server 使用不同的端口,则 with 还将在单独的随机端口上启动 Management Server。webEnvironment = WebEnvironment.RANDOM_PORT |
9.3.1. 检测 Web 应用程序类型
如果 Spring MVC 可用,则配置基于常规 MVC 的应用程序上下文。 如果您只有 Spring WebFlux,我们将检测到该情况并配置基于 WebFlux 的应用程序上下文。
如果两者都存在,则 Spring MVC 优先。
如果要在此场景中测试反应式 Web 应用程序,则必须设置以下属性:spring.main.web-application-type
@SpringBootTest(properties = "spring.main.web-application-type=reactive")
class MyWebFluxTests {
// ...
}
@SpringBootTest(properties = ["spring.main.web-application-type=reactive"])
class MyWebFluxTests {
// ...
}
9.3.2. 检测测试配置
如果您熟悉 Spring Test Framework,则可能习惯于使用来指定要加载的 Spring。
或者,您可能经常在测试中使用嵌套类。@ContextConfiguration(classes=…)
@Configuration
@Configuration
在测试 Spring Boot 应用程序时,这通常不是必需的。
Spring Boot 的 Comments 会在您未明确定义主配置时自动搜索主配置。@*Test
搜索算法从包含测试的包开始工作,直到找到带有 或 注释的类。
只要你以合理的方式构建你的代码,你的主要配置通常会被找到。@SpringBootApplication
@SpringBootConfiguration
如果使用 test annotation 来测试应用程序的更具体部分,则应避免在 main 方法的 application 类上添加特定于特定区域的配置设置。 的底层组件扫描配置定义了用于确保切片按预期工作的排除过滤器。
如果您在 -annotated 类上使用 explicit 指令,请注意这些过滤器将被禁用。
如果您正在使用切片,则应再次定义它们。 |
如果要自定义主配置,可以使用嵌套类。
与嵌套类不同,嵌套类将用于代替应用程序的主配置,而嵌套类则除了应用程序的主配置外,还使用嵌套类。@TestConfiguration
@Configuration
@TestConfiguration
Spring 的测试框架在测试之间缓存应用程序上下文。 因此,只要您的测试共享相同的配置(无论它是如何发现的),加载上下文的潜在耗时过程只会发生一次。 |
9.3.3. 使用测试配置 main 方法
通常,发现的测试配置将是您的主要 。
在大多数结构良好的应用程序中,此 configuration 类还将包括用于启动应用程序的方法。@SpringBootTest
@SpringBootApplication
main
例如,以下是典型 Spring Boot 应用程序的非常常见的 Code Pattern:
@SpringBootApplication
public class MyApplication {
public static void main(String[] args) {
SpringApplication.run(MyApplication.class, args);
}
}
@SpringBootApplication
class MyApplication
fun main(args: Array<String>) {
runApplication<MyApplication>(*args)
}
在上面的示例中,该方法除了委托 .
但是,可以使用更复杂的方法在调用 .main
SpringApplication.run
main
SpringApplication.run
例如,下面是一个更改横幅模式并设置其他配置文件的应用程序:
@SpringBootApplication
public class MyApplication {
public static void main(String[] args) {
SpringApplication application = new SpringApplication(MyApplication.class);
application.setBannerMode(Banner.Mode.OFF);
application.setAdditionalProfiles("myprofile");
application.run(args);
}
}
@SpringBootApplication
class MyApplication
fun main(args: Array<String>) {
runApplication<MyApplication>(*args) {
setBannerMode(Banner.Mode.OFF)
setAdditionalProfiles("myprofile")
}
}
由于方法中的自定义可能会影响结果,因此您可能还希望使用该方法在测试中创建 used。
默认情况下,不会调用您的方法,而是直接使用类本身来创建main
ApplicationContext
main
ApplicationContext
@SpringBootTest
main
ApplicationContext
如果要更改此行为,可以将 的属性更改为 或 。
设置为 时,如果找不到方法,则测试将失败。
设置为 时,如果可用,则使用方法,否则将使用标准加载机制。useMainMethod
@SpringBootTest
UseMainMethod.ALWAYS
UseMainMethod.WHEN_AVAILABLE
ALWAYS
main
WHEN_AVAILABLE
main
例如,以下测试将调用 of 的方法以创建 .
如果 main 方法设置了其他配置文件,则这些配置文件将在启动时处于活动状态。main
MyApplication
ApplicationContext
ApplicationContext
@SpringBootTest(useMainMethod = UseMainMethod.ALWAYS)
class MyApplicationTests {
@Test
void exampleTest() {
// ...
}
}
@SpringBootTest(useMainMethod = UseMainMethod.ALWAYS)
class MyApplicationTests {
@Test
fun exampleTest() {
// ...
}
}
9.3.4. 排除测试配置
如果您的应用程序使用组件扫描(例如,如果您使用 或 ),您可能会发现仅为特定测试创建的顶级配置类会意外地随处可见。@SpringBootApplication
@ComponentScan
正如我们之前看到的,可以用于测试的内部类来自定义主要配置。 也可以在顶级类上使用。这样做表示不应通过扫描来选取该类。
然后,您可以在需要的地方显式导入该类,如以下示例所示:@TestConfiguration
@TestConfiguration
@SpringBootTest
@Import(MyTestsConfiguration.class)
class MyTests {
@Test
void exampleTest() {
// ...
}
}
@SpringBootTest
@Import(MyTestsConfiguration::class)
class MyTests {
@Test
fun exampleTest() {
// ...
}
}
如果直接使用 (即,不是通过 ),则需要向 它注册 。
有关详细信息,请参阅 Javadoc。@ComponentScan @SpringBootApplication TypeExcludeFilter |
导入的 an 比 inner-class 更早处理,而 imported 将在通过组件扫描找到任何配置之前处理。
一般来说,这种 Sequences 的差异没有明显的影响,但是如果你依赖于 bean 覆盖,则需要注意这一点。@TestConfiguration @TestConfiguration @TestConfiguration |
9.3.5. 使用应用程序参数
如果您的应用程序需要参数,则可以
使用 attribute 注入它们。@SpringBootTest
args
@SpringBootTest(args = "--app.test=one")
class MyApplicationArgumentTests {
@Test
void applicationArgumentsPopulated(@Autowired ApplicationArguments args) {
assertThat(args.getOptionNames()).containsOnly("app.test");
assertThat(args.getOptionValues("app.test")).containsOnly("one");
}
}
@SpringBootTest(args = ["--app.test=one"])
class MyApplicationArgumentTests {
@Test
fun applicationArgumentsPopulated(@Autowired args: ApplicationArguments) {
assertThat(args.optionNames).containsOnly("app.test")
assertThat(args.getOptionValues("app.test")).containsOnly("one")
}
}
9.3.6. 使用 Mock 环境进行测试
默认情况下,不会启动服务器,而是设置用于测试 Web 端点的模拟环境。@SpringBootTest
借助 Spring MVC,我们可以使用 MockMvc
或 查询我们的 Web 端点,如以下示例所示:WebTestClient
@SpringBootTest
@AutoConfigureMockMvc
class MyMockMvcTests {
@Test
void testWithMockMvc(@Autowired MockMvc mvc) throws Exception {
mvc.perform(get("/")).andExpect(status().isOk()).andExpect(content().string("Hello World"));
}
// If Spring WebFlux is on the classpath, you can drive MVC tests with a WebTestClient
@Test
void testWithWebTestClient(@Autowired WebTestClient webClient) {
webClient
.get().uri("/")
.exchange()
.expectStatus().isOk()
.expectBody(String.class).isEqualTo("Hello World");
}
}
@SpringBootTest
@AutoConfigureMockMvc
class MyMockMvcTests {
@Test
fun testWithMockMvc(@Autowired mvc: MockMvc) {
mvc.perform(MockMvcRequestBuilders.get("/")).andExpect(MockMvcResultMatchers.status().isOk)
.andExpect(MockMvcResultMatchers.content().string("Hello World"))
}
// If Spring WebFlux is on the classpath, you can drive MVC tests with a WebTestClient
@Test
fun testWithWebTestClient(@Autowired webClient: WebTestClient) {
webClient
.get().uri("/")
.exchange()
.expectStatus().isOk
.expectBody<String>().isEqualTo("Hello World")
}
}
如果您只想关注 Web 图层,而不开始 完整 ,请考虑改用 @WebMvcTest 。ApplicationContext |
使用 Spring WebFlux 端点,你可以使用WebTestClient
,如以下示例所示:
@SpringBootTest
@AutoConfigureWebTestClient
class MyMockWebTestClientTests {
@Test
void exampleTest(@Autowired WebTestClient webClient) {
webClient
.get().uri("/")
.exchange()
.expectStatus().isOk()
.expectBody(String.class).isEqualTo("Hello World");
}
}
@SpringBootTest
@AutoConfigureWebTestClient
class MyMockWebTestClientTests {
@Test
fun exampleTest(@Autowired webClient: WebTestClient) {
webClient
.get().uri("/")
.exchange()
.expectStatus().isOk
.expectBody<String>().isEqualTo("Hello World")
}
}
在模拟环境中进行测试通常比使用完整的 servlet 容器运行更快。 但是,由于模拟发生在 Spring MVC 层,因此无法直接使用 MockMvc 测试依赖于较低级别 servlet 容器行为的代码。 例如, Spring Boot 的错误处理基于 servlet 容器提供的“错误页面”支持。 这意味着,虽然您可以测试 MVC 层是否按预期引发和处理异常,但不能直接测试是否呈现了特定的自定义错误页面。 如果需要测试这些较低级别的问题,可以按照下一节所述启动完全运行的服务器。 |
9.3.7. 使用正在运行的服务器进行测试
如果您需要启动一个完全运行的服务器,我们建议您使用随机端口。
如果使用 ,则每次运行测试时都会随机选择一个可用端口。@SpringBootTest(webEnvironment=WebEnvironment.RANDOM_PORT)
该 annotation 可用于将实际使用的端口注入到测试中。
为方便起见,需要对已启动的服务器进行 REST 调用的测试还可以自动连接 WebTestClient
,该客户端解析到正在运行的服务器的相对链接,并附带一个用于验证响应的专用 API,如以下示例所示:@LocalServerPort
@SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT)
class MyRandomPortWebTestClientTests {
@Test
void exampleTest(@Autowired WebTestClient webClient) {
webClient
.get().uri("/")
.exchange()
.expectStatus().isOk()
.expectBody(String.class).isEqualTo("Hello World");
}
}
@SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT)
class MyRandomPortWebTestClientTests {
@Test
fun exampleTest(@Autowired webClient: WebTestClient) {
webClient
.get().uri("/")
.exchange()
.expectStatus().isOk
.expectBody<String>().isEqualTo("Hello World")
}
}
WebTestClient 还可以与模拟环境一起使用,无需运行服务器,方法是使用 .@AutoConfigureWebTestClient |
此设置需要在 classpath 上。
如果你不能或不打算添加 webflux,Spring Boot 还提供了一个工具:spring-webflux
TestRestTemplate
@SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT)
class MyRandomPortTestRestTemplateTests {
@Test
void exampleTest(@Autowired TestRestTemplate restTemplate) {
String body = restTemplate.getForObject("/", String.class);
assertThat(body).isEqualTo("Hello World");
}
}
@SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT)
class MyRandomPortTestRestTemplateTests {
@Test
fun exampleTest(@Autowired restTemplate: TestRestTemplate) {
val body = restTemplate.getForObject("/", String::class.java)
assertThat(body).isEqualTo("Hello World")
}
}
9.3.8. 自定义 WebTestClient
要自定义 Bean,请配置 Bean。
任何此类 bean 都使用用于创建 的 调用。WebTestClient
WebTestClientBuilderCustomizer
WebTestClient.Builder
WebTestClient
9.3.9. 使用 JMX
由于测试上下文框架会缓存上下文,因此默认情况下会禁用 JMX,以防止相同的组件在同一域上注册。
如果此类测试需要访问 ,则还要考虑将其标记为 dirty:MBeanServer
@SpringBootTest(properties = "spring.jmx.enabled=true")
@DirtiesContext
class MyJmxTests {
@Autowired
private MBeanServer mBeanServer;
@Test
void exampleTest() {
assertThat(this.mBeanServer.getDomains()).contains("java.lang");
// ...
}
}
@SpringBootTest(properties = ["spring.jmx.enabled=true"])
@DirtiesContext
class MyJmxTests(@Autowired val mBeanServer: MBeanServer) {
@Test
fun exampleTest() {
assertThat(mBeanServer.domains).contains("java.lang")
// ...
}
}
9.3.10. 使用观察
如果使用 注释切片测试,它会自动配置一个 .@AutoConfigureObservability
ObservationRegistry
9.3.11. 使用度量
无论您的 Classpath 如何,在使用 时,不会自动配置计量注册表(内存中支持的除外)。@SpringBootTest
如果需要将指标导出到其他后端作为集成测试的一部分,请使用 .@AutoConfigureObservability
如果使用 注释切片测试,则会自动配置内存中的 .
注释不支持在切片测试中导出数据。@AutoConfigureObservability
MeterRegistry
@AutoConfigureObservability
9.3.12. 使用跟踪
无论您的 Classpath 如何,使用 时,不会自动配置报告数据的跟踪组件。@SpringBootTest
如果您需要这些组件作为集成测试的一部分,请使用 .@AutoConfigureObservability
如果您已创建自己的报表组件(例如自定义 或 ),并且不希望它们在测试中处于活动状态,则可以使用注释来禁用它们。SpanExporter
SpanHandler
@ConditionalOnEnabledTracing
如果使用 注释切片测试,则会自动配置 no-op 。
注释不支持在切片测试中导出数据。@AutoConfigureObservability
Tracer
@AutoConfigureObservability
9.3.13. Mocking 和 Spying Bean
运行测试时,有时需要模拟应用程序上下文中的某些组件。 例如,您可能在某些远程服务上有一个在开发过程中不可用的门面。 当您想要模拟在真实环境中可能难以触发的故障时,模拟也很有用。
Spring Boot 包含一个注释,可用于为 .
您可以使用注释添加新的 bean 或替换单个现有的 bean 定义。
注解可以直接用于测试类、测试中的字段或类和字段。
当用于字段时,创建的 mock 的实例也会被注入。
Mock bean 在每个测试方法之后都会自动重置。@MockBean
ApplicationContext
@Configuration
如果您的测试使用 Spring Boot 的 test annotations(例如 ),则会自动启用此功能。
要将此功能与不同的 arrangement 一起使用,必须显式添加侦听器,如以下示例所示: Java
Kotlin
|
以下示例将现有 bean 替换为 mock 实现:RemoteService
@SpringBootTest
class MyTests {
@Autowired
private Reverser reverser;
@MockBean
private RemoteService remoteService;
@Test
void exampleTest() {
given(this.remoteService.getValue()).willReturn("spring");
String reverse = this.reverser.getReverseValue(); // Calls injected RemoteService
assertThat(reverse).isEqualTo("gnirps");
}
}
@SpringBootTest
class MyTests(@Autowired val reverser: Reverser, @MockBean val remoteService: RemoteService) {
@Test
fun exampleTest() {
given(remoteService.value).willReturn("spring")
val reverse = reverser.reverseValue // Calls injected RemoteService
assertThat(reverse).isEqualTo("gnirps")
}
}
@MockBean 不能用于模拟在应用程序上下文刷新期间执行的 bean 的行为。
执行测试时,应用程序上下文刷新已完成,配置模拟行为为时已晚。
在这种情况下,我们建议使用一种方法来创建和配置 mock。@Bean |
此外,您还可以使用 Mockito 包装任何现有 bean。
有关完整详细信息,请参阅 Javadoc。@SpyBean
spy
虽然 Spring 的测试框架在测试之间缓存应用程序上下文,并为共享相同配置的测试重用上下文,但使用或影响缓存键,这很可能会增加上下文的数量。@MockBean @SpyBean |
如果用于使用按名称引用参数的方法监视 bean,则必须使用 .
这可确保在监视 Bean 后,参数名称可用于缓存基础结构。@SpyBean @Cacheable -parameters |
当您用于监视由 Spring 代理的 bean 时,在某些情况下可能需要删除 Spring 的代理,例如,当使用 or 设置期望时。
用于执行此操作。@SpyBean given when AopTestUtils.getTargetObject(yourProxiedSpy) |
9.3.14. 自动配置的测试
Spring Boot 的自动配置系统适用于应用程序,但有时对于测试来说可能有点太多了。 通常,仅加载测试应用程序的 “slice” 所需的配置部分会有所帮助。 例如,你可能想要测试 Spring MVC 控制器是否正确映射 URL,并且你不想在这些测试中涉及数据库调用,或者你可能想要测试 JPA 实体,并且你对这些测试运行时的 Web 层不感兴趣。
该模块包含许多可用于自动配置此类 “切片” 的 Comments。
它们中的每一个都以类似的方式工作,提供一个用于加载 的注释,以及一个或多个可用于自定义自动配置设置的注释。spring-boot-test-autoconfigure
@…Test
ApplicationContext
@AutoConfigure…
每个 slice 将组件扫描限制为适当的组件,并加载一组非常受限的 auto-configuration classes。
如果需要排除其中一个,大多数 Comments 都会提供一个属性。
或者,您也可以使用 .@…Test excludeAutoConfiguration @ImportAutoConfiguration#exclude |
不支持在一个测试中使用多个 Comments 来包含多个 “slice”。
如果您需要多个 “切片”,请选择其中一个注释并手动包含其他 “切片” 的注释。@…Test @…Test @AutoConfigure… |
也可以将 Comments 与标准注释一起使用。
如果您对 “切片” 应用程序不感兴趣,但想要一些自动配置的测试 bean,则可以使用此组合。@AutoConfigure… @SpringBootTest |
9.3.15. 自动配置的 JSON 测试
要测试对象 JSON 序列化和反序列化是否按预期工作,您可以使用注释。 自动配置可用的受支持 JSON 映射器,该映射器可以是以下库之一:@JsonTest
@JsonTest
-
Jackson,任何豆子和任何 Jackson s
ObjectMapper
@JsonComponent
Module
-
Gson
-
Jsonb
附录中提供了 启用的自动配置列表。@JsonTest |
如果需要配置自动配置的元素,则可以使用 annotation.@AutoConfigureJsonTesters
Spring Boot 包括基于 AssertJ 的帮助程序,这些帮助程序与 JSONAssert 和 JsonPath 库一起使用,以检查 JSON 是否按预期显示。
、、 、 和 类可以分别用于 Jackson、 Gson、 Jsonb 和 Strings。
测试类上的任何帮助程序字段都可以在使用 .
以下示例显示了 Jackson 的 test 类:JacksonTester
GsonTester
JsonbTester
BasicJsonTester
@Autowired
@JsonTest
@JsonTest
class MyJsonTests {
@Autowired
private JacksonTester<VehicleDetails> json;
@Test
void serialize() throws Exception {
VehicleDetails details = new VehicleDetails("Honda", "Civic");
// Assert against a `.json` file in the same package as the test
assertThat(this.json.write(details)).isEqualToJson("expected.json");
// Or use JSON path based assertions
assertThat(this.json.write(details)).hasJsonPathStringValue("@.make");
assertThat(this.json.write(details)).extractingJsonPathStringValue("@.make").isEqualTo("Honda");
}
@Test
void deserialize() throws Exception {
String content = "{\"make\":\"Ford\",\"model\":\"Focus\"}";
assertThat(this.json.parse(content)).isEqualTo(new VehicleDetails("Ford", "Focus"));
assertThat(this.json.parseObject(content).getMake()).isEqualTo("Ford");
}
}
@JsonTest
class MyJsonTests(@Autowired val json: JacksonTester<VehicleDetails>) {
@Test
fun serialize() {
val details = VehicleDetails("Honda", "Civic")
// Assert against a `.json` file in the same package as the test
assertThat(json.write(details)).isEqualToJson("expected.json")
// Or use JSON path based assertions
assertThat(json.write(details)).hasJsonPathStringValue("@.make")
assertThat(json.write(details)).extractingJsonPathStringValue("@.make").isEqualTo("Honda")
}
@Test
fun deserialize() {
val content = "{\"make\":\"Ford\",\"model\":\"Focus\"}"
assertThat(json.parse(content)).isEqualTo(VehicleDetails("Ford", "Focus"))
assertThat(json.parseObject(content).make).isEqualTo("Ford")
}
}
JSON 帮助程序类也可以直接在标准单元测试中使用。
为此,如果不使用 ,请在您的方法中调用帮助程序的方法。initFields @Before @JsonTest |
如果使用 Spring Boot 基于 AssertJ 的帮助程序在给定 JSON 路径上的数字值上断言,则可能无法使用,具体取决于类型。
相反,您可以使用 AssertJ 来断言该值与给定条件匹配。
例如,以下示例断言实际数字是接近偏移量 的 float 值。isEqualTo
satisfies
0.15
0.01
@Test
void someTest() throws Exception {
SomeObject value = new SomeObject(0.152f);
assertThat(this.json.write(value)).extractingJsonPathNumberValue("@.test.numberValue")
.satisfies((number) -> assertThat(number.floatValue()).isCloseTo(0.15f, within(0.01f)));
}
@Test
fun someTest() {
val value = SomeObject(0.152f)
assertThat(json.write(value)).extractingJsonPathNumberValue("@.test.numberValue")
.satisfies(ThrowingConsumer { number ->
assertThat(number.toFloat()).isCloseTo(0.15f, within(0.01f))
})
}
9.3.16. 自动配置的 Spring MVC 测试
要测试 Spring MVC 控制器是否按预期工作,请使用 Comments。 自动配置 Spring MVC 基础结构,并将扫描的 bean 限制为 、 和 。
使用 Comments 时,不会扫描 Regular 和 bean。 可用于包含 bean。@WebMvcTest
@WebMvcTest
@Controller
@ControllerAdvice
@JsonComponent
Converter
GenericConverter
Filter
HandlerInterceptor
WebMvcConfigurer
WebMvcRegistrations
HandlerMethodArgumentResolver
@Component
@ConfigurationProperties
@WebMvcTest
@EnableConfigurationProperties
@ConfigurationProperties
附录中提供了启用的自动配置设置的列表。@WebMvcTest |
如果需要注册额外的组件,例如 Jackson ,则可以在测试中使用导入其他配置类。Module @Import |
通常,仅限于单个控制器,并结合使用,为所需的协作者提供模拟实现。@WebMvcTest
@MockBean
@WebMvcTest
还自动配置 .
Mock MVC 提供了一种强大的方法来快速测试 MVC 控制器,而无需启动完整的 HTTP 服务器。MockMvc
您还可以在非 - (例如 ) 中自动配置,方法是使用 .
以下示例使用 :MockMvc @WebMvcTest @SpringBootTest @AutoConfigureMockMvc MockMvc |
@WebMvcTest(UserVehicleController.class)
class MyControllerTests {
@Autowired
private MockMvc mvc;
@MockBean
private UserVehicleService userVehicleService;
@Test
void testExample() throws Exception {
given(this.userVehicleService.getVehicleDetails("sboot"))
.willReturn(new VehicleDetails("Honda", "Civic"));
this.mvc.perform(get("/sboot/vehicle").accept(MediaType.TEXT_PLAIN))
.andExpect(status().isOk())
.andExpect(content().string("Honda Civic"));
}
}
@WebMvcTest(UserVehicleController::class)
class MyControllerTests(@Autowired val mvc: MockMvc) {
@MockBean
lateinit var userVehicleService: UserVehicleService
@Test
fun testExample() {
given(userVehicleService.getVehicleDetails("sboot"))
.willReturn(VehicleDetails("Honda", "Civic"))
mvc.perform(MockMvcRequestBuilders.get("/sboot/vehicle").accept(MediaType.TEXT_PLAIN))
.andExpect(MockMvcResultMatchers.status().isOk)
.andExpect(MockMvcResultMatchers.content().string("Honda Civic"))
}
}
如果需要配置自动配置的元素(例如,何时应应用 servlet 过滤器),则可以在 Comments 中使用属性。@AutoConfigureMockMvc |
如果使用 HtmlUnit 和 Selenium,则自动配置还会提供 HtmlUnit bean 和/或 Selenium bean。
下面的示例使用 HtmlUnit:WebClient
WebDriver
@WebMvcTest(UserVehicleController.class)
class MyHtmlUnitTests {
@Autowired
private WebClient webClient;
@MockBean
private UserVehicleService userVehicleService;
@Test
void testExample() throws Exception {
given(this.userVehicleService.getVehicleDetails("sboot")).willReturn(new VehicleDetails("Honda", "Civic"));
HtmlPage page = this.webClient.getPage("/sboot/vehicle.html");
assertThat(page.getBody().getTextContent()).isEqualTo("Honda Civic");
}
}
@WebMvcTest(UserVehicleController::class)
class MyHtmlUnitTests(@Autowired val webClient: WebClient) {
@MockBean
lateinit var userVehicleService: UserVehicleService
@Test
fun testExample() {
given(userVehicleService.getVehicleDetails("sboot")).willReturn(VehicleDetails("Honda", "Civic"))
val page = webClient.getPage<HtmlPage>("/sboot/vehicle.html")
assertThat(page.body.textContent).isEqualTo("Honda Civic")
}
}
默认情况下, Spring Boot 将 bean 放在一个特殊的“范围”中,以确保驱动程序在每次测试后退出并注入新实例。
如果您不希望此行为,可以添加到您的定义中。WebDriver @Scope("singleton") WebDriver @Bean |
Spring Boot 创建的范围将替换任何用户定义的同名范围。
如果您定义自己的范围,您可能会发现它在使用 .webDriver webDriver @WebMvcTest |
如果您在 Classpath 上安装了 Spring Security,则还将扫描 bean。
您可以使用 Spring Security 的测试支持,而不是完全禁用此类测试的安全性。
有关如何使用 Spring Security 支持的更多详细信息,请参阅此 howto.html操作方法部分。@WebMvcTest
WebSecurityConfigurer
MockMvc
有时编写 Spring MVC 测试是不够的;Spring Boot 可以帮助您使用实际服务器运行完整的端到端测试。 |
9.3.17. 自动配置的 Spring WebFlux 测试
要测试 Spring WebFlux 控制器是否按预期工作,你可以使用 Comments。 自动配置 Spring WebFlux 基础结构,并将扫描的 bean 限制为 、 和 。
使用 Comments 时,不会扫描 Regular 和 bean。 可用于包含 bean。@WebFluxTest
@WebFluxTest
@Controller
@ControllerAdvice
@JsonComponent
Converter
GenericConverter
WebFilter
WebFluxConfigurer
@Component
@ConfigurationProperties
@WebFluxTest
@EnableConfigurationProperties
@ConfigurationProperties
附录中提供了 启用的自动配置列表。@WebFluxTest |
如果需要注册额外的组件,例如 Jackson ,则可以使用 on your test 导入其他配置类。Module @Import |
通常,仅限于单个控制器,并与 annotation 结合使用,为所需的协作者提供 mock 实现。@WebFluxTest
@MockBean
@WebFluxTest
还自动配置了 WebTestClient
,它提供了一种强大的方法来快速测试 WebFlux 控制器,而无需启动完整的 HTTP 服务器。
您还可以在非 - (例如 ) 中自动配置,方法是使用 .
以下示例显示了一个同时使用 和 a 的类:WebTestClient @WebFluxTest @SpringBootTest @AutoConfigureWebTestClient @WebFluxTest WebTestClient |
@WebFluxTest(UserVehicleController.class)
class MyControllerTests {
@Autowired
private WebTestClient webClient;
@MockBean
private UserVehicleService userVehicleService;
@Test
void testExample() {
given(this.userVehicleService.getVehicleDetails("sboot"))
.willReturn(new VehicleDetails("Honda", "Civic"));
this.webClient.get().uri("/sboot/vehicle").accept(MediaType.TEXT_PLAIN).exchange()
.expectStatus().isOk()
.expectBody(String.class).isEqualTo("Honda Civic");
}
}
@WebFluxTest(UserVehicleController::class)
class MyControllerTests(@Autowired val webClient: WebTestClient) {
@MockBean
lateinit var userVehicleService: UserVehicleService
@Test
fun testExample() {
given(userVehicleService.getVehicleDetails("sboot"))
.willReturn(VehicleDetails("Honda", "Civic"))
webClient.get().uri("/sboot/vehicle").accept(MediaType.TEXT_PLAIN).exchange()
.expectStatus().isOk
.expectBody<String>().isEqualTo("Honda Civic")
}
}
此设置仅受 WebFlux 应用程序支持,因为在模拟的 Web 应用程序中使用目前仅适用于 WebFlux。WebTestClient |
@WebFluxTest 无法检测通过 Functional Web 框架注册的路由。
要在上下文中测试 bean,请考虑使用 或 using 导入您自己。RouterFunction RouterFunction @Import @SpringBootTest |
@WebFluxTest 无法检测注册为 类型 的自定义安全配置 。
要将其包含在测试中,您需要导入使用 或 using 注册 bean 的配置。@Bean SecurityWebFilterChain @Import @SpringBootTest |
有时编写 Spring WebFlux 测试是不够的;Spring Boot 可以帮助您使用实际服务器运行完整的端到端测试。 |
9.3.18. 自动配置的 Spring GraphQL 测试
Spring GraphQL 提供了一个专用的测试支持模块;您需要将其添加到您的项目中:
<dependencies>
<dependency>
<groupId>org.springframework.graphql</groupId>
<artifactId>spring-graphql-test</artifactId>
<scope>test</scope>
</dependency>
<!-- Unless already present in the compile scope -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-webflux</artifactId>
<scope>test</scope>
</dependency>
</dependencies>
dependencies {
testImplementation("org.springframework.graphql:spring-graphql-test")
// Unless already present in the implementation configuration
testImplementation("org.springframework.boot:spring-boot-starter-webflux")
}
此测试模块附带 GraphQlTester。
测试器在测试中被大量使用,因此请务必熟悉它的使用。
有一些变体,Spring Boot 将根据测试类型自动配置它们:GraphQlTester
-
在服务器端执行测试,没有客户端或传输
ExecutionGraphQlServiceTester
-
使用连接到服务器的客户端执行测试,无论是否有实时服务器
HttpGraphQlTester
Spring Boot 可帮助您使用注释测试 Spring GraphQL 控制器。 自动配置 Spring GraphQL 基础设施,无需任何传输或服务器。
这会将扫描的 bean 限制为 、 和 。
使用 Comments 时,不会扫描 Regular 和 bean。 可用于包含 bean。@GraphQlTest
@GraphQlTest
@Controller
RuntimeWiringConfigurer
JsonComponent
Converter
GenericConverter
DataFetcherExceptionResolver
Instrumentation
GraphQlSourceBuilderCustomizer
@Component
@ConfigurationProperties
@GraphQlTest
@EnableConfigurationProperties
@ConfigurationProperties
附录中提供了 启用的自动配置列表。@GraphQlTest |
通常,仅限于一组控制器,并与 annotation 结合使用,为所需的协作者提供 mock 实现。@GraphQlTest
@MockBean
@GraphQlTest(GreetingController.class)
class GreetingControllerTests {
@Autowired
private GraphQlTester graphQlTester;
@Test
void shouldGreetWithSpecificName() {
this.graphQlTester.document("{ greeting(name: \"Alice\") } ")
.execute()
.path("greeting")
.entity(String.class)
.isEqualTo("Hello, Alice!");
}
@Test
void shouldGreetWithDefaultName() {
this.graphQlTester.document("{ greeting } ")
.execute()
.path("greeting")
.entity(String.class)
.isEqualTo("Hello, Spring!");
}
}
@GraphQlTest(GreetingController::class)
internal class GreetingControllerTests {
@Autowired
lateinit var graphQlTester: GraphQlTester
@Test
fun shouldGreetWithSpecificName() {
graphQlTester.document("{ greeting(name: \"Alice\") } ").execute().path("greeting").entity(String::class.java)
.isEqualTo("Hello, Alice!")
}
@Test
fun shouldGreetWithDefaultName() {
graphQlTester.document("{ greeting } ").execute().path("greeting").entity(String::class.java)
.isEqualTo("Hello, Spring!")
}
}
@SpringBootTest
测试是完整的集成测试,涉及整个应用程序。
当使用随机或定义的端口时,将配置一个实时服务器并自动提供一个 bean,以便您可以使用它来测试您的服务器。
配置 MOCK 环境后,您还可以通过使用以下 Comments 来请求测试类 Bean :HttpGraphQlTester
HttpGraphQlTester
@AutoConfigureHttpGraphQlTester
@AutoConfigureHttpGraphQlTester
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.MOCK)
class GraphQlIntegrationTests {
@Test
void shouldGreetWithSpecificName(@Autowired HttpGraphQlTester graphQlTester) {
HttpGraphQlTester authenticatedTester = graphQlTester.mutate()
.webTestClient((client) -> client.defaultHeaders((headers) -> headers.setBasicAuth("admin", "ilovespring")))
.build();
authenticatedTester.document("{ greeting(name: \"Alice\") } ")
.execute()
.path("greeting")
.entity(String.class)
.isEqualTo("Hello, Alice!");
}
}
@AutoConfigureHttpGraphQlTester
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.MOCK)
class GraphQlIntegrationTests {
@Test
fun shouldGreetWithSpecificName(@Autowired graphQlTester: HttpGraphQlTester) {
val authenticatedTester = graphQlTester.mutate()
.webTestClient { client: WebTestClient.Builder ->
client.defaultHeaders { headers: HttpHeaders ->
headers.setBasicAuth("admin", "ilovespring")
}
}.build()
authenticatedTester.document("{ greeting(name: \"Alice\") } ").execute()
.path("greeting").entity(String::class.java).isEqualTo("Hello, Alice!")
}
}
9.3.19. 自动配置的 Data Cassandra 测试
可用于测试 Cassandra 应用程序。
默认情况下,它配置一个 ,扫描类,并配置 Spring Data Cassandra 存储库。
使用 Comments 时,不会扫描 Regular 和 bean。 可用于包含 bean。
(有关将 Cassandra 与 Spring Boot 结合使用的更多信息,请参阅“data.html”。@DataCassandraTest
CassandraTemplate
@Table
@Component
@ConfigurationProperties
@DataCassandraTest
@EnableConfigurationProperties
@ConfigurationProperties
附录中提供了启用的自动配置设置的列表。@DataCassandraTest |
以下示例显示了在 Spring Boot 中使用 Cassandra 测试的典型设置:
@DataCassandraTest
class MyDataCassandraTests {
@Autowired
private SomeRepository repository;
}
@DataCassandraTest
class MyDataCassandraTests(@Autowired val repository: SomeRepository)
9.3.20. 自动配置的数据 Couchbase 测试
您可以使用它来测试 Couchbase 应用程序。
默认情况下,它配置或扫描类,并配置 Spring Data Couchbase 存储库。
使用 Comments 时,不会扫描 Regular 和 bean。 可用于包含 bean。
(有关将 Couchbase 与 Spring Boot 结合使用的更多信息,请参阅本章前面的“data.html”。@DataCouchbaseTest
CouchbaseTemplate
ReactiveCouchbaseTemplate
@Document
@Component
@ConfigurationProperties
@DataCouchbaseTest
@EnableConfigurationProperties
@ConfigurationProperties
附录中提供了启用的自动配置设置的列表。@DataCouchbaseTest |
以下示例显示了在 Spring Boot 中使用 Couchbase 测试的典型设置:
@DataCouchbaseTest
class MyDataCouchbaseTests {
@Autowired
private SomeRepository repository;
// ...
}
@DataCouchbaseTest
class MyDataCouchbaseTests(@Autowired val repository: SomeRepository) {
// ...
}
9.3.21. 自动配置的数据 Elasticsearch 测试
您可以使用它来测试 Elasticsearch 应用程序。
默认情况下,它会配置一个 、 扫描类,并配置 Spring Data Elasticsearch 存储库。
使用 Comments 时,不会扫描 Regular 和 bean。 可用于包含 bean。
(有关将 Elasticsearch 与 Spring Boot 结合使用的更多信息,请参阅本章前面的“data.html”。@DataElasticsearchTest
ElasticsearchRestTemplate
@Document
@Component
@ConfigurationProperties
@DataElasticsearchTest
@EnableConfigurationProperties
@ConfigurationProperties
附录中提供了启用的自动配置设置的列表。@DataElasticsearchTest |
以下示例显示了在 Spring Boot 中使用 Elasticsearch 测试的典型设置:
@DataElasticsearchTest
class MyDataElasticsearchTests {
@Autowired
private SomeRepository repository;
// ...
}
@DataElasticsearchTest
class MyDataElasticsearchTests(@Autowired val repository: SomeRepository) {
// ...
}
9.3.22. 自动配置的数据 JPA 测试
您可以使用注释来测试 JPA 应用程序。
默认情况下,它会扫描类并配置 Spring Data JPA 存储库。
如果 Classpath 上有嵌入式数据库可用,则它也会配置一个。
默认情况下,通过将属性设置为 SQL 查询来记录 SQL 查询。
可以使用 annotation 的属性禁用此功能。@DataJpaTest
@Entity
spring.jpa.show-sql
true
showSql
使用 Comments 时,不会扫描 Regular 和 bean。 可用于包含 bean。@Component
@ConfigurationProperties
@DataJpaTest
@EnableConfigurationProperties
@ConfigurationProperties
附录中提供了启用的自动配置设置的列表。@DataJpaTest |
默认情况下,数据 JPA 测试是事务性的,并在每次测试结束时回滚。 有关更多详细信息,请参阅 Spring Framework Reference Documentation 中的相关部分。 如果这不是您想要的,则可以禁用测试或整个类的事务 management,如下所示:
@DataJpaTest
@Transactional(propagation = Propagation.NOT_SUPPORTED)
class MyNonTransactionalTests {
// ...
}
@DataJpaTest
@Transactional(propagation = Propagation.NOT_SUPPORTED)
class MyNonTransactionalTests {
// ...
}
数据 JPA 测试还可以注入 TestEntityManager
Bean,它为专为测试设计的标准 JPA 提供了替代方案。EntityManager
TestEntityManager 也可以通过添加 .
执行此操作时,请确保您的测试在事务中运行,例如通过添加测试类或方法。@AutoConfigureTestEntityManager @Transactional |
如果您需要,也可以使用 A。
以下示例显示了正在使用的注释:JdbcTemplate
@DataJpaTest
@DataJpaTest
class MyRepositoryTests {
@Autowired
private TestEntityManager entityManager;
@Autowired
private UserRepository repository;
@Test
void testExample() {
this.entityManager.persist(new User("sboot", "1234"));
User user = this.repository.findByUsername("sboot");
assertThat(user.getUsername()).isEqualTo("sboot");
assertThat(user.getEmployeeNumber()).isEqualTo("1234");
}
}
@DataJpaTest
class MyRepositoryTests(@Autowired val entityManager: TestEntityManager, @Autowired val repository: UserRepository) {
@Test
fun testExample() {
entityManager.persist(User("sboot", "1234"))
val user = repository.findByUsername("sboot")
assertThat(user?.username).isEqualTo("sboot")
assertThat(user?.employeeNumber).isEqualTo("1234")
}
}
内存嵌入式数据库通常适用于测试,因为它们速度很快且不需要任何安装。
但是,如果您希望对实际数据库运行测试,则可以使用 Comments,如以下示例所示:@AutoConfigureTestDatabase
@DataJpaTest
@AutoConfigureTestDatabase(replace = Replace.NONE)
class MyRepositoryTests {
// ...
}
@DataJpaTest
@AutoConfigureTestDatabase(replace = AutoConfigureTestDatabase.Replace.NONE)
class MyRepositoryTests {
// ...
}
9.3.23. 自动配置的 JDBC 测试
@JdbcTest
类似于,但适用于只需要 Spring Data JDBC 而不使用 Spring Data JDBC 的测试。
默认情况下,它会配置一个内存中嵌入式数据库和一个 .
使用 Comments 时,不会扫描 Regular 和 bean。 可用于包含 bean。@DataJpaTest
DataSource
JdbcTemplate
@Component
@ConfigurationProperties
@JdbcTest
@EnableConfigurationProperties
@ConfigurationProperties
附录中提供了 启用的自动配置列表。@JdbcTest |
默认情况下,JDBC 测试是事务性的,并在每个测试结束时回滚。 有关更多详细信息,请参阅 Spring Framework Reference Documentation 中的相关部分。 如果这不是您想要的,则可以为测试或整个类禁用事务管理,如下所示:
@JdbcTest
@Transactional(propagation = Propagation.NOT_SUPPORTED)
class MyTransactionalTests {
}
@JdbcTest
@Transactional(propagation = Propagation.NOT_SUPPORTED)
class MyTransactionalTests
如果您希望测试针对实际数据库运行,则可以像使用 Comments 一样使用 Comments。
(请参阅“自动配置的数据 JPA 测试”)@AutoConfigureTestDatabase
@DataJpaTest
9.3.24. 自动配置的数据 JDBC 测试
@DataJdbcTest
类似于,但适用于使用 Spring Data JDBC 存储库的测试。
默认情况下,它配置内存中嵌入式数据库、 和 Spring Data JDBC 存储库。
使用 Comments 时仅扫描子类,不扫描 regular 和 bean。 可用于包含 bean。@JdbcTest
JdbcTemplate
AbstractJdbcConfiguration
@DataJdbcTest
@Component
@ConfigurationProperties
@EnableConfigurationProperties
@ConfigurationProperties
附录中提供了 启用的自动配置列表。@DataJdbcTest |
默认情况下,数据 JDBC 测试是事务性的,并在每个测试结束时回滚。 有关更多详细信息,请参阅 Spring Framework Reference Documentation 中的相关部分。 如果这不是您想要的,则可以禁用测试或整个测试类的事务管理,如 JDBC 示例所示。
如果您希望测试针对实际数据库运行,则可以像使用 Comments 一样使用 Comments。
(请参阅“自动配置的数据 JPA 测试”)@AutoConfigureTestDatabase
@DataJpaTest
9.3.25. 自动配置的数据 R2DBC 测试
@DataR2dbcTest
类似于,但适用于使用 Spring Data R2DBC 存储库的测试。
默认情况下,它配置内存中的嵌入式数据库、 和 Spring Data R2DBC 存储库。
使用 Comments 时,不会扫描 Regular 和 bean。 可用于包含 bean。@DataJdbcTest
R2dbcEntityTemplate
@Component
@ConfigurationProperties
@DataR2dbcTest
@EnableConfigurationProperties
@ConfigurationProperties
附录中提供了 启用的自动配置列表。@DataR2dbcTest |
默认情况下,Data R2DBC 测试不是事务性的。
如果您希望测试针对实际数据库运行,则可以像使用 Comments 一样使用 Comments。
(请参阅“自动配置的数据 JPA 测试”)@AutoConfigureTestDatabase
@DataJpaTest
9.3.26. 自动配置的 jOOQ 测试
您可以以与 jOOQ 相关的测试类似的方式使用。
由于 jOOQ 严重依赖与数据库架构相对应的基于 Java 的架构,因此使用 existing。
如果要将其替换为内存中数据库,可以使用 覆盖这些设置。
(有关在 Spring Boot 中使用 jOOQ 的更多信息,请参阅“data.html”。
使用 Comments 时,不会扫描 Regular 和 bean。 可用于包含 bean。@JooqTest
@JdbcTest
DataSource
@AutoConfigureTestDatabase
@Component
@ConfigurationProperties
@JooqTest
@EnableConfigurationProperties
@ConfigurationProperties
附录中提供了 启用的自动配置列表。@JooqTest |
@JooqTest
配置 .
以下示例显示了正在使用的注释:DSLContext
@JooqTest
@JooqTest
class MyJooqTests {
@Autowired
private DSLContext dslContext;
// ...
}
@JooqTest
class MyJooqTests(@Autowired val dslContext: DSLContext) {
// ...
}
默认情况下,JOOQ 测试是事务性的,并且在每个测试结束时回滚。 如果这不是您想要的,则可以禁用测试或整个测试类的事务管理,如 JDBC 示例所示。
9.3.27. 自动配置的数据 MongoDB 测试
可用于测试 MongoDB 应用程序。
默认情况下,它会配置一个 、 扫描类,并配置 Spring Data MongoDB 存储库。
使用 Comments 时,不会扫描 Regular 和 bean。 可用于包含 bean。
(有关将 MongoDB 与 Spring Boot 结合使用的更多信息,请参阅“data.html”。@DataMongoTest
MongoTemplate
@Document
@Component
@ConfigurationProperties
@DataMongoTest
@EnableConfigurationProperties
@ConfigurationProperties
附录中提供了启用的自动配置设置的列表。@DataMongoTest |
下面的类显示了正在使用的 Annotation:@DataMongoTest
@DataMongoTest
class MyDataMongoDbTests {
@Autowired
private MongoTemplate mongoTemplate;
// ...
}
@DataMongoTest
class MyDataMongoDbTests(@Autowired val mongoTemplate: MongoTemplate) {
// ...
}
9.3.28. 自动配置的数据 Neo4j 测试
您可以使用它来测试 Neo4j 应用程序。
默认情况下,它会扫描类并配置 Spring Data Neo4j 存储库。
使用 Comments 时,不会扫描 Regular 和 bean。 可用于包含 bean。
(有关将 Neo4J 与 Spring Boot 结合使用的更多信息,请参阅“data.html”。@DataNeo4jTest
@Node
@Component
@ConfigurationProperties
@DataNeo4jTest
@EnableConfigurationProperties
@ConfigurationProperties
附录中提供了启用的自动配置设置的列表。@DataNeo4jTest |
以下示例显示了在 Spring Boot 中使用 Neo4J 测试的典型设置:
@DataNeo4jTest
class MyDataNeo4jTests {
@Autowired
private SomeRepository repository;
// ...
}
@DataNeo4jTest
class MyDataNeo4jTests(@Autowired val repository: SomeRepository) {
// ...
}
默认情况下,Data Neo4j 测试是事务性的,并在每次测试结束时回滚。 有关更多详细信息,请参阅 Spring Framework Reference Documentation 中的相关部分。 如果这不是您想要的,则可以为测试或整个类禁用事务管理,如下所示:
@DataNeo4jTest
@Transactional(propagation = Propagation.NOT_SUPPORTED)
class MyDataNeo4jTests {
}
@DataNeo4jTest
@Transactional(propagation = Propagation.NOT_SUPPORTED)
class MyDataNeo4jTests
反应式访问不支持事务性测试。
如果使用此样式,则必须如上所述配置测试。@DataNeo4jTest |
9.3.29. 自动配置的数据 Redis 测试
您可以使用它来测试 Redis 应用程序。
默认情况下,它会扫描类并配置 Spring Data Redis 存储库。
使用 Comments 时,不会扫描 Regular 和 bean。 可用于包含 bean。
(有关将 Redis 与 Spring Boot 结合使用的更多信息,请参阅“data.html”。@DataRedisTest
@RedisHash
@Component
@ConfigurationProperties
@DataRedisTest
@EnableConfigurationProperties
@ConfigurationProperties
附录中提供了启用的自动配置设置的列表。@DataRedisTest |
以下示例显示了正在使用的注释:@DataRedisTest
@DataRedisTest
class MyDataRedisTests {
@Autowired
private SomeRepository repository;
// ...
}
@DataRedisTest
class MyDataRedisTests(@Autowired val repository: SomeRepository) {
// ...
}
9.3.30. 自动配置的数据 LDAP 测试
可用于测试 LDAP 应用程序。
默认情况下,它会配置内存中的嵌入式 LDAP(如果可用),配置,扫描类,并配置 Spring Data LDAP 存储库。
使用 Comments 时,不会扫描 Regular 和 bean。 可用于包含 bean。
(有关将 LDAP 与 Spring Boot 结合使用的更多信息,请参阅“data.html”。@DataLdapTest
LdapTemplate
@Entry
@Component
@ConfigurationProperties
@DataLdapTest
@EnableConfigurationProperties
@ConfigurationProperties
附录中提供了启用的自动配置设置的列表。@DataLdapTest |
以下示例显示了正在使用的注释:@DataLdapTest
@DataLdapTest
class MyDataLdapTests {
@Autowired
private LdapTemplate ldapTemplate;
// ...
}
@DataLdapTest
class MyDataLdapTests(@Autowired val ldapTemplate: LdapTemplate) {
// ...
}
内存中嵌入式 LDAP 通常适用于测试,因为它速度很快,并且不需要任何开发人员安装。 但是,如果您希望针对实际 LDAP 服务器运行测试,则应排除嵌入式 LDAP 自动配置,如以下示例所示:
@DataLdapTest(excludeAutoConfiguration = EmbeddedLdapAutoConfiguration.class)
class MyDataLdapTests {
// ...
}
@DataLdapTest(excludeAutoConfiguration = [EmbeddedLdapAutoConfiguration::class])
class MyDataLdapTests {
// ...
}
9.3.31. 自动配置的 REST 客户端
您可以使用注释来测试 REST 客户端。
默认情况下,它会自动配置 Jackson、GSON 和 Jsonb 支持,配置 a 和 a ,并添加对 .
使用 Comments 时,不会扫描 Regular 和 bean。 可用于包含 bean。@RestClientTest
RestTemplateBuilder
RestClient.Builder
MockRestServiceServer
@Component
@ConfigurationProperties
@RestClientTest
@EnableConfigurationProperties
@ConfigurationProperties
附录中提供了启用的自动配置设置的列表。@RestClientTest |
要测试的特定 bean 应使用 的 or 属性指定。value
components
@RestClientTest
当在待测试的 bean 中使用 a 并在构建 时被调用时,则应从预期中省略根 URI,如以下示例所示:RestTemplateBuilder
RestTemplateBuilder.rootUri(String rootUri)
RestTemplate
MockRestServiceServer
@RestClientTest(RemoteVehicleDetailsService.class)
class MyRestTemplateServiceTests {
@Autowired
private RemoteVehicleDetailsService service;
@Autowired
private MockRestServiceServer server;
@Test
void getVehicleDetailsWhenResultIsSuccessShouldReturnDetails() {
this.server.expect(requestTo("/greet/details")).andRespond(withSuccess("hello", MediaType.TEXT_PLAIN));
String greeting = this.service.callRestService();
assertThat(greeting).isEqualTo("hello");
}
}
@RestClientTest(RemoteVehicleDetailsService::class)
class MyRestTemplateServiceTests(
@Autowired val service: RemoteVehicleDetailsService,
@Autowired val server: MockRestServiceServer) {
@Test
fun getVehicleDetailsWhenResultIsSuccessShouldReturnDetails() {
server.expect(MockRestRequestMatchers.requestTo("/greet/details"))
.andRespond(MockRestResponseCreators.withSuccess("hello", MediaType.TEXT_PLAIN))
val greeting = service.callRestService()
assertThat(greeting).isEqualTo("hello")
}
}
在待测试的bean中使用 a 时,或者在使用 a without calling 时,必须在预期中使用完整的 URI,如以下示例所示:RestClient.Builder
RestTemplateBuilder
rootUri(String rootURI)
MockRestServiceServer
@RestClientTest(RemoteVehicleDetailsService.class)
class MyRestClientServiceTests {
@Autowired
private RemoteVehicleDetailsService service;
@Autowired
private MockRestServiceServer server;
@Test
void getVehicleDetailsWhenResultIsSuccessShouldReturnDetails() {
this.server.expect(requestTo("https://example.com/greet/details"))
.andRespond(withSuccess("hello", MediaType.TEXT_PLAIN));
String greeting = this.service.callRestService();
assertThat(greeting).isEqualTo("hello");
}
}
@RestClientTest(RemoteVehicleDetailsService::class)
class MyRestClientServiceTests(
@Autowired val service: RemoteVehicleDetailsService,
@Autowired val server: MockRestServiceServer) {
@Test
fun getVehicleDetailsWhenResultIsSuccessShouldReturnDetails() {
server.expect(MockRestRequestMatchers.requestTo("https://example.com/greet/details"))
.andRespond(MockRestResponseCreators.withSuccess("hello", MediaType.TEXT_PLAIN))
val greeting = service.callRestService()
assertThat(greeting).isEqualTo("hello")
}
}
9.3.32. 自动配置的 Spring REST Docs 测试
您可以使用注释在带有 Mock MVC、REST Assure 或 WebTestClient 的测试中使用 Spring REST Docs。
它消除了 Spring REST Docs 中对 JUnit 扩展的需求。@AutoConfigureRestDocs
@AutoConfigureRestDocs
可用于覆盖默认输出目录(如果您使用的是 Maven 或 Gradle)。
它还可用于配置出现在任何记录的 URI 中的主机、方案和端口。target/generated-snippets
build/generated-snippets
使用模拟 MVC 自动配置的 Spring REST 文档测试
@AutoConfigureRestDocs
自定义 Bean 以在测试基于 servlet 的 Web 应用程序时使用 Spring REST Docs。
你可以通过使用它来注入它,并在测试中使用它,就像你通常使用 Mock MVC 和 Spring REST Docs 时一样,如以下示例所示:MockMvc
@Autowired
@WebMvcTest(UserController.class)
@AutoConfigureRestDocs
class MyUserDocumentationTests {
@Autowired
private MockMvc mvc;
@Test
void listUsers() throws Exception {
this.mvc.perform(get("/users").accept(MediaType.TEXT_PLAIN))
.andExpect(status().isOk())
.andDo(document("list-users"));
}
}
@WebMvcTest(UserController::class)
@AutoConfigureRestDocs
class MyUserDocumentationTests(@Autowired val mvc: MockMvc) {
@Test
fun listUsers() {
mvc.perform(MockMvcRequestBuilders.get("/users").accept(MediaType.TEXT_PLAIN))
.andExpect(MockMvcResultMatchers.status().isOk)
.andDo(MockMvcRestDocumentation.document("list-users"))
}
}
如果您需要对 Spring REST Docs 配置的控制比 的属性提供更多的控制,则可以使用 bean,如以下示例所示:@AutoConfigureRestDocs
RestDocsMockMvcConfigurationCustomizer
@TestConfiguration(proxyBeanMethods = false)
public class MyRestDocsConfiguration implements RestDocsMockMvcConfigurationCustomizer {
@Override
public void customize(MockMvcRestDocumentationConfigurer configurer) {
configurer.snippets().withTemplateFormat(TemplateFormats.markdown());
}
}
@TestConfiguration(proxyBeanMethods = false)
class MyRestDocsConfiguration : RestDocsMockMvcConfigurationCustomizer {
override fun customize(configurer: MockMvcRestDocumentationConfigurer) {
configurer.snippets().withTemplateFormat(TemplateFormats.markdown())
}
}
如果你想利用 Spring REST Docs 对参数化输出目录的支持,你可以创建一个 bean。
使用此结果处理程序的 auto-configuration 调用,从而导致每个调用自动生成默认代码段。
以下示例显示了一个 being defined:RestDocumentationResultHandler
alwaysDo
MockMvc
RestDocumentationResultHandler
@TestConfiguration(proxyBeanMethods = false)
public class MyResultHandlerConfiguration {
@Bean
public RestDocumentationResultHandler restDocumentation() {
return MockMvcRestDocumentation.document("{method-name}");
}
}
@TestConfiguration(proxyBeanMethods = false)
class MyResultHandlerConfiguration {
@Bean
fun restDocumentation(): RestDocumentationResultHandler {
return MockMvcRestDocumentation.document("{method-name}")
}
}
使用 WebTestClient 自动配置的 Spring REST Docs 测试
@AutoConfigureRestDocs
也可以在测试反应式 Web 应用程序时使用。
你可以通过使用 using 来注入它,并在测试中使用它,就像你通常使用 Spring REST Docs 时一样,如以下示例所示:WebTestClient
@Autowired
@WebFluxTest
@WebFluxTest
@AutoConfigureRestDocs
class MyUsersDocumentationTests {
@Autowired
private WebTestClient webTestClient;
@Test
void listUsers() {
this.webTestClient
.get().uri("/")
.exchange()
.expectStatus()
.isOk()
.expectBody()
.consumeWith(document("list-users"));
}
}
@WebFluxTest
@AutoConfigureRestDocs
class MyUsersDocumentationTests(@Autowired val webTestClient: WebTestClient) {
@Test
fun listUsers() {
webTestClient
.get().uri("/")
.exchange()
.expectStatus()
.isOk
.expectBody()
.consumeWith(WebTestClientRestDocumentation.document("list-users"))
}
}
如果您需要对 Spring REST Docs 配置的控制比 的属性提供更多的控制,则可以使用 bean,如以下示例所示:@AutoConfigureRestDocs
RestDocsWebTestClientConfigurationCustomizer
@TestConfiguration(proxyBeanMethods = false)
public class MyRestDocsConfiguration implements RestDocsWebTestClientConfigurationCustomizer {
@Override
public void customize(WebTestClientRestDocumentationConfigurer configurer) {
configurer.snippets().withEncoding("UTF-8");
}
}
@TestConfiguration(proxyBeanMethods = false)
class MyRestDocsConfiguration : RestDocsWebTestClientConfigurationCustomizer {
override fun customize(configurer: WebTestClientRestDocumentationConfigurer) {
configurer.snippets().withEncoding("UTF-8")
}
}
如果你想利用 Spring REST Docs 对参数化输出目录的支持,你可以使用 a 为每个实体交换结果配置一个使用者。
以下示例显示了这样一个被定义:WebTestClientBuilderCustomizer
WebTestClientBuilderCustomizer
@TestConfiguration(proxyBeanMethods = false)
public class MyWebTestClientBuilderCustomizerConfiguration {
@Bean
public WebTestClientBuilderCustomizer restDocumentation() {
return (builder) -> builder.entityExchangeResultConsumer(document("{method-name}"));
}
}
@TestConfiguration(proxyBeanMethods = false)
class MyWebTestClientBuilderCustomizerConfiguration {
@Bean
fun restDocumentation(): WebTestClientBuilderCustomizer {
return WebTestClientBuilderCustomizer { builder: WebTestClient.Builder ->
builder.entityExchangeResultConsumer(
WebTestClientRestDocumentation.document("{method-name}")
)
}
}
}
使用 REST Assured 自动配置的 Spring REST Docs 测试
@AutoConfigureRestDocs
使预配置为使用 Spring REST Docs 的 bean 可用于您的测试。
你可以通过使用它来注入它,并在测试中使用它,就像你通常使用 REST Assured 和 Spring REST Docs 时一样,如以下示例所示:RequestSpecification
@Autowired
@SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT)
@AutoConfigureRestDocs
class MyUserDocumentationTests {
@Test
void listUsers(@Autowired RequestSpecification documentationSpec, @LocalServerPort int port) {
given(documentationSpec)
.filter(document("list-users"))
.when()
.port(port)
.get("/")
.then().assertThat()
.statusCode(is(200));
}
}
@SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT)
@AutoConfigureRestDocs
class MyUserDocumentationTests {
@Test
fun listUsers(@Autowired documentationSpec: RequestSpecification?, @LocalServerPort port: Int) {
RestAssured.given(documentationSpec)
.filter(RestAssuredRestDocumentation.document("list-users"))
.`when`()
.port(port)["/"]
.then().assertThat()
.statusCode(Matchers.`is`(200))
}
}
如果您需要对 Spring REST Docs 配置的控制比 的属性 提供更多的控制,则可以使用 bean,如以下示例所示:@AutoConfigureRestDocs
RestDocsRestAssuredConfigurationCustomizer
@TestConfiguration(proxyBeanMethods = false)
public class MyRestDocsConfiguration implements RestDocsRestAssuredConfigurationCustomizer {
@Override
public void customize(RestAssuredRestDocumentationConfigurer configurer) {
configurer.snippets().withTemplateFormat(TemplateFormats.markdown());
}
}
@TestConfiguration(proxyBeanMethods = false)
class MyRestDocsConfiguration : RestDocsRestAssuredConfigurationCustomizer {
override fun customize(configurer: RestAssuredRestDocumentationConfigurer) {
configurer.snippets().withTemplateFormat(TemplateFormats.markdown())
}
}
9.3.33. 自动配置的 Spring Web 服务测试
自动配置的 Spring Web 服务客户端测试
可用于测试使用 Spring Web Services 项目调用 Web 服务的应用程序。
默认情况下,它会配置一个模拟 bean 并自动自定义您的 .
(有关将 Web 服务与 Spring Boot 结合使用的更多信息,请参阅“io.html”。@WebServiceClientTest
WebServiceServer
WebServiceTemplateBuilder
附录中提供了启用的自动配置设置的列表。@WebServiceClientTest |
以下示例显示了正在使用的注释:@WebServiceClientTest
@WebServiceClientTest(SomeWebService.class)
class MyWebServiceClientTests {
@Autowired
private MockWebServiceServer server;
@Autowired
private SomeWebService someWebService;
@Test
void mockServerCall() {
this.server
.expect(payload(new StringSource("<request/>")))
.andRespond(withPayload(new StringSource("<response><status>200</status></response>")));
assertThat(this.someWebService.test())
.extracting(Response::getStatus)
.isEqualTo(200);
}
}
@WebServiceClientTest(SomeWebService::class)
class MyWebServiceClientTests(@Autowired val server: MockWebServiceServer, @Autowired val someWebService: SomeWebService) {
@Test
fun mockServerCall() {
server
.expect(RequestMatchers.payload(StringSource("<request/>")))
.andRespond(ResponseCreators.withPayload(StringSource("<response><status>200</status></response>")))
assertThat(this.someWebService.test()).extracting(Response::status).isEqualTo(200)
}
}
自动配置的 Spring Web 服务服务器测试
可用于测试使用 Spring Web Services 项目实现 Web 服务的应用程序。
默认情况下,它配置一个可用于调用 Web 服务端点的 Bean。
(有关将 Web 服务与 Spring Boot 结合使用的更多信息,请参阅“io.html”。@WebServiceServerTest
MockWebServiceClient
附录中提供了启用的自动配置设置的列表。@WebServiceServerTest |
以下示例显示了正在使用的注释:@WebServiceServerTest
@WebServiceServerTest(ExampleEndpoint.class)
class MyWebServiceServerTests {
@Autowired
private MockWebServiceClient client;
@Test
void mockServerCall() {
this.client
.sendRequest(RequestCreators.withPayload(new StringSource("<ExampleRequest/>")))
.andExpect(ResponseMatchers.payload(new StringSource("<ExampleResponse>42</ExampleResponse>")));
}
}
@WebServiceServerTest(ExampleEndpoint::class)
class MyWebServiceServerTests(@Autowired val client: MockWebServiceClient) {
@Test
fun mockServerCall() {
client
.sendRequest(RequestCreators.withPayload(StringSource("<ExampleRequest/>")))
.andExpect(ResponseMatchers.payload(StringSource("<ExampleResponse>42</ExampleResponse>")))
}
}
9.3.34. 其他自动配置和切片
每个切片都提供一个或多个注释,即定义应作为切片的一部分包含的自动配置。
通过创建自定义注释或添加到测试,可以逐个测试添加其他自动配置,如以下示例所示:@AutoConfigure…
@AutoConfigure…
@ImportAutoConfiguration
@JdbcTest
@ImportAutoConfiguration(IntegrationAutoConfiguration.class)
class MyJdbcTests {
}
@JdbcTest
@ImportAutoConfiguration(IntegrationAutoConfiguration::class)
class MyJdbcTests
确保不要使用常规 Comments 来导入自动配置,因为它们是由 Spring Boot 以特定方式处理的。@Import |
或者,可以通过在存储的文件中注册切片注释来为切片注释的任何使用添加其他自动配置,如以下示例所示:META-INF/spring
com.example.IntegrationAutoConfiguration
在此示例中,在每个带有 .com.example.IntegrationAutoConfiguration
@JdbcTest
您可以在此文件中使用 Comments。# |
切片或批注可以通过这种方式进行自定义,只要它使用 .@AutoConfigure… @ImportAutoConfiguration |
9.3.35. 用户配置和切片
如果您以合理的方式构建代码,则默认情况下会使用您的类作为测试的配置。@SpringBootApplication
因此,重要的是不要在应用程序的主类中混淆特定于其特定功能区域的配置设置。
假设您使用的是 Spring Data MongoDB,您依赖于它的自动配置,并且您已启用审计。
您可以按如下方式定义 your :@SpringBootApplication
@SpringBootApplication
@EnableMongoAuditing
public class MyApplication {
// ...
}
@SpringBootApplication
@EnableMongoAuditing
class MyApplication {
// ...
}
因为这个类是测试的源配置,所以任何切片测试实际上都会尝试启用 Mongo 审计,这绝对不是你想要的。
建议的方法是将特定于区域的配置移动到与应用程序位于同一级别的单独类,如以下示例所示:@Configuration
@Configuration(proxyBeanMethods = false)
@EnableMongoAuditing
public class MyMongoConfiguration {
// ...
}
@Configuration(proxyBeanMethods = false)
@EnableMongoAuditing
class MyMongoConfiguration {
// ...
}
根据应用程序的复杂程度,您可能有一个用于自定义的类,或者每个域区域有一个类。
后一种方法允许您在其中一个测试中启用它(如有必要),并使用 annotation。
有关何时可能需要为 slice 测试启用特定类的更多详细信息,请参阅此操作方法部分。@Configuration @Import @Configuration |
测试切片从扫描中排除类。
例如,对于 a ,以下配置将不会在测试切片加载的应用程序上下文中包含给定的 bean:@Configuration
@WebMvcTest
WebMvcConfigurer
@Configuration(proxyBeanMethods = false)
public class MyWebConfiguration {
@Bean
public WebMvcConfigurer testConfigurer() {
return new WebMvcConfigurer() {
// ...
};
}
}
@Configuration(proxyBeanMethods = false)
class MyWebConfiguration {
@Bean
fun testConfigurer(): WebMvcConfigurer {
return object : WebMvcConfigurer {
// ...
}
}
}
但是,下面的配置将导致测试切片加载自定义。WebMvcConfigurer
@Component
public class MyWebMvcConfigurer implements WebMvcConfigurer {
// ...
}
@Component
class MyWebMvcConfigurer : WebMvcConfigurer {
// ...
}
另一个混淆的来源是 Classpath scanning。 假设您以合理的方式构建了代码,但需要扫描其他包。 您的应用程序可能类似于以下代码:
@SpringBootApplication
@ComponentScan({ "com.example.app", "com.example.another" })
public class MyApplication {
// ...
}
@SpringBootApplication
@ComponentScan("com.example.app", "com.example.another")
class MyApplication {
// ...
}
这样做可以有效地覆盖默认的组件 scan 指令,其副作用是扫描这两个包,而不管你选择了哪个 slice。
例如,a 似乎突然扫描了应用程序的组件和用户配置。
同样,将 custom 指令移动到单独的类是解决此问题的好方法。@DataJpaTest
如果这不是您的选项,则可以在测试的层次结构中创建一个 somewhere ,以便改用它。
或者,您可以为测试指定一个源,这将禁用查找默认源的行为。@SpringBootConfiguration |
9.3.36. 使用 Spock 测试 Spring Boot 应用程序
Spock 2.2 或更高版本可用于测试 Spring Boot 应用程序。
为此,请将对 Spock 模块版本的依赖项添加到应用程序的构建中。 将 Spring 的测试框架集成到 Spock 中。
有关更多详细信息,请参阅 Spock 的 Spring 模块的文档。-groovy-4.0
spock-spring
spock-spring
9.4. 测试容器
Testcontainers 库提供了一种管理在 Docker 容器中运行的服务的方法。 它与 JUnit 集成,允许您编写一个测试类,该类可以在任何测试运行之前启动容器。 Testcontainers 对于编写与实际后端服务(如 MySQL、MongoDB、Cassandra 等)通信的集成测试特别有用。
Testcontainers 可以在 Spring Boot 测试中使用,如下所示:
@Testcontainers
@SpringBootTest
class MyIntegrationTests {
@Container
static Neo4jContainer<?> neo4j = new Neo4jContainer<>("neo4j:5");
@Test
void myTest() {
// ...
}
}
@Testcontainers
@SpringBootTest
class MyIntegrationTests {
@Test
fun myTest() {
// ...
}
companion object {
@Container
val neo4j = Neo4jContainer("neo4j:5")
}
}
这将在运行任何测试之前启动运行 Neo4j(如果 Docker 在本地运行)的 docker 容器。 在大多数情况下,您需要配置应用程序以连接到容器中运行的服务。
9.4.1. 服务连接
服务连接是与任何远程服务的连接。 Spring Boot 的自动配置可以使用服务连接的详细信息,并使用它们来建立与远程服务的连接。 执行此操作时,连接详细信息优先于任何与连接相关的配置属性。
使用 Testcontainers 时,可以通过在 test 类中注释 container 字段,为容器中运行的服务自动创建连接详细信息。
@Testcontainers
@SpringBootTest
class MyIntegrationTests {
@Container
@ServiceConnection
static Neo4jContainer<?> neo4j = new Neo4jContainer<>("neo4j:5");
@Test
void myTest() {
// ...
}
}
@Testcontainers
@SpringBootTest
class MyIntegrationTests {
@Test
fun myTest() {
// ...
}
companion object {
@Container
@ServiceConnection
val neo4j = Neo4jContainer("neo4j:5")
}
}
多亏了,上述配置允许应用程序中与 Neo4j 相关的 bean 与在 Testcontainers 管理的 Docker 容器中运行的 Neo4j 进行通信。
这是通过自动定义一个 bean 来完成的,然后由 Neo4j 自动配置使用,覆盖任何与连接相关的配置属性。@ServiceConnection
Neo4jConnectionDetails
您需要将模块添加为测试依赖项,以便将服务连接与 Testcontainers 一起使用。spring-boot-testcontainers |
服务连接注释由向 注册的类处理。
A 可以基于特定的子类或 Docker 镜像名称创建 Bean。ContainerConnectionDetailsFactory
spring.factories
ContainerConnectionDetailsFactory
ConnectionDetails
Container
jar 中提供了以下服务连接工厂:spring-boot-testcontainers
连接详细信息 | 匹配时间 |
---|---|
|
名为 “symptoma/activemq” 的容器 |
|
类型的容器 |
|
类型的容器 |
|
类型的容器 |
|
类型的容器 |
|
类型的容器 |
|
类型为 或 |
|
类型的容器 |
|
类型的容器 |
|
类型的容器 |
|
名为 “otel/opentelemetry-collector-contrib” 的容器 |
|
名为 “otel/opentelemetry-collector-contrib” 的容器 |
|
类型的容器 |
|
类型为 、 、 、 或 |
|
类型的容器 |
|
名为 “redis” 的容器 |
|
名为 “openzipkin/zipkin” 的容器 |
默认情况下,将为给定的 .
例如,a 将同时创建 和 。 如果只想创建适用类型的子集,则可以使用 的属性 . |
默认情况下,用于获取用于查找连接详细信息的名称。
Docker 镜像名称的存储库部分会忽略任何注册表和版本。
只要 Spring Boot 能够获取 instance 的实例,这就可以工作,例如上例中所示的字段。Container.getDockerImageName().getRepository()
Container
static
如果你正在使用方法, Spring Boot 不会调用 bean 方法来获取 Docker 镜像名称,因为这会导致急切的初始化问题。
相反,使用 bean 方法的返回类型来找出应该使用哪个连接详细信息。
只要你使用的是类型化容器,这就可以正常工作,例如 或。
如果你正在使用 ,例如与 Redis 一起使用,这将停止工作,如以下示例所示:@Bean
Neo4jContainer
RabbitMQContainer
GenericContainer
@TestConfiguration(proxyBeanMethods = false)
public class MyRedisConfiguration {
@Bean
@ServiceConnection(name = "redis")
public GenericContainer<?> redisContainer() {
return new GenericContainer<>("redis:7");
}
}
@TestConfiguration(proxyBeanMethods = false)
class MyRedisConfiguration {
@Bean
@ServiceConnection(name = "redis")
fun redisContainer(): GenericContainer<*> {
return GenericContainer("redis:7")
}
}
Spring Boot 无法判断使用了哪个容器镜像,因此必须使用属性from来提供该提示。GenericContainer
name
@ServiceConnection
您还可以使用属性 of 来覆盖将使用的连接详细信息,例如在使用自定义图像时。
如果您使用的是 Docker 镜像 ,您将用于确保已创建。name
@ServiceConnection
registry.mycompany.com/mirror/myredis
@ServiceConnection(name="redis")
RedisConnectionDetails
9.4.2. 动态属性
与服务连接相比,一个稍微详细但更灵活的替代方案是 。
static 方法允许向 Spring Environment 添加动态属性值。@DynamicPropertySource
@DynamicPropertySource
@Testcontainers
@SpringBootTest
class MyIntegrationTests {
@Container
static Neo4jContainer<?> neo4j = new Neo4jContainer<>("neo4j:5");
@Test
void myTest() {
// ...
}
@DynamicPropertySource
static void neo4jProperties(DynamicPropertyRegistry registry) {
registry.add("spring.neo4j.uri", neo4j::getBoltUrl);
}
}
@Testcontainers
@SpringBootTest
class MyIntegrationTests {
@Test
fun myTest() {
// ...
}
companion object {
@Container
val neo4j = Neo4jContainer("neo4j:5")
@DynamicPropertySource
fun neo4jProperties(registry: DynamicPropertyRegistry) {
registry.add("spring.neo4j.uri") { neo4j.boltUrl }
}
}
}
上述配置允许应用程序中与 Neo4j 相关的 bean 与在 Testcontainers 管理的 Docker 容器内运行的 Neo4j 进行通信。
9.5. 测试工具
在测试应用程序时通常有用的一些测试实用程序类打包为 的一部分。spring-boot
9.5.1. ConfigDataApplicationContextInitializer
ConfigDataApplicationContextInitializer
是可以应用于测试以加载 Spring Boot 文件。
当您不需要 提供的完整功能集时,可以使用它,如以下示例所示:ApplicationContextInitializer
application.properties
@SpringBootTest
@ContextConfiguration(classes = Config.class, initializers = ConfigDataApplicationContextInitializer.class)
class MyConfigFileTests {
// ...
}
@ContextConfiguration(classes = [Config::class], initializers = [ConfigDataApplicationContextInitializer::class])
class MyConfigFileTests {
// ...
}
单独使用不支持注射。
它唯一的工作是确保将文件加载到 Spring 的 .
要获得支持,您需要额外配置 a 或使用 ,这将为您自动配置 一个。ConfigDataApplicationContextInitializer @Value("${…}") application.properties Environment @Value PropertySourcesPlaceholderConfigurer @SpringBootTest |
9.5.2. TestPropertyValues
TestPropertyValues
用于快速向 或 添加属性。
您可以使用字符串调用它,如下所示:ConfigurableEnvironment
ConfigurableApplicationContext
key=value
class MyEnvironmentTests {
@Test
void testPropertySources() {
MockEnvironment environment = new MockEnvironment();
TestPropertyValues.of("org=Spring", "name=Boot").applyTo(environment);
assertThat(environment.getProperty("name")).isEqualTo("Boot");
}
}
class MyEnvironmentTests {
@Test
fun testPropertySources() {
val environment = MockEnvironment()
TestPropertyValues.of("org=Spring", "name=Boot").applyTo(environment)
assertThat(environment.getProperty("name")).isEqualTo("Boot")
}
}
9.5.3. 输出捕获
OutputCapture
是可用于捕获和输出的 JUnit。
要使用它,请将 add 和 inject 作为参数添加到您的测试类构造函数或测试方法中,如下所示:Extension
System.out
System.err
@ExtendWith(OutputCaptureExtension.class)
CapturedOutput
@ExtendWith(OutputCaptureExtension.class)
class MyOutputCaptureTests {
@Test
void testName(CapturedOutput output) {
System.out.println("Hello World!");
assertThat(output).contains("World");
}
}
@ExtendWith(OutputCaptureExtension::class)
class MyOutputCaptureTests {
@Test
fun testName(output: CapturedOutput?) {
println("Hello World!")
assertThat(output).contains("World")
}
}
9.5.4. TestRestTemplate
TestRestTemplate
是 Spring 的便捷替代方案,在集成测试中很有用。
你可以获取一个 vanilla 模板或发送 Basic HTTP 身份验证的模板(带有用户名和密码)。
在任一情况下,模板都是容错的。
这意味着它的行为对测试友好,不会在 4xx 和 5xx 错误上引发异常。
相反,可以通过返回的 state code 及其 status code 来检测此类错误。RestTemplate
ResponseEntity
Spring Framework 5.0 提供了一个适用于 WebFlux 集成测试以及 WebFlux 和 MVC 端到端测试的新功能。
它为断言提供了流畅的 API,这与 .WebTestClient TestRestTemplate |
建议使用 Apache HTTP 客户端(版本 5.1 或更高版本),但并非强制性要求。
如果你在 Classpath 上有它,则通过适当地配置 Client 端来响应。
如果您确实使用 Apache 的 HTTP 客户端,则会启用一些其他测试友好功能:TestRestTemplate
-
不遵循重定向(因此您可以断言响应位置)。
-
Cookie 将被忽略(因此模板是无状态的)。
TestRestTemplate
可以直接在集成测试中实例化,如以下示例所示:
class MyTests {
private final TestRestTemplate template = new TestRestTemplate();
@Test
void testRequest() {
ResponseEntity<String> headers = this.template.getForEntity("https://myhost.example.com/example", String.class);
assertThat(headers.getHeaders().getLocation()).hasHost("other.example.com");
}
}
class MyTests {
private val template = TestRestTemplate()
@Test
fun testRequest() {
val headers = template.getForEntity("https://myhost.example.com/example", String::class.java)
assertThat(headers.headers.location).hasHost("other.example.com")
}
}
或者,如果将注释与 或 一起使用,则可以注入完全配置的注释并开始使用它。
如有必要,可以通过 Bean 应用其他定制。
任何未指定主机和端口的 URL 都会自动连接到嵌入式服务器,如以下示例所示:@SpringBootTest
WebEnvironment.RANDOM_PORT
WebEnvironment.DEFINED_PORT
TestRestTemplate
RestTemplateBuilder
@SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT)
class MySpringBootTests {
@Autowired
private TestRestTemplate template;
@Test
void testRequest() {
HttpHeaders headers = this.template.getForEntity("/example", String.class).getHeaders();
assertThat(headers.getLocation()).hasHost("other.example.com");
}
@TestConfiguration(proxyBeanMethods = false)
static class RestTemplateBuilderConfiguration {
@Bean
RestTemplateBuilder restTemplateBuilder() {
return new RestTemplateBuilder().setConnectTimeout(Duration.ofSeconds(1))
.setReadTimeout(Duration.ofSeconds(1));
}
}
}
@SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT)
class MySpringBootTests(@Autowired val template: TestRestTemplate) {
@Test
fun testRequest() {
val headers = template.getForEntity("/example", String::class.java).headers
assertThat(headers.location).hasHost("other.example.com")
}
@TestConfiguration(proxyBeanMethods = false)
internal class RestTemplateBuilderConfiguration {
@Bean
fun restTemplateBuilder(): RestTemplateBuilder {
return RestTemplateBuilder().setConnectTimeout(Duration.ofSeconds(1))
.setReadTimeout(Duration.ofSeconds(1))
}
}
}
10. Docker Compose 支持
Docker Compose 是一种流行的技术,可用于为应用程序所需的服务定义和管理多个容器。
通常在应用程序旁边创建一个文件,用于定义和配置服务容器。compose.yml
Docker Compose 的典型工作流是运行,在应用程序上运行,将其连接到已启动的服务,然后在完成后运行。docker compose up
docker compose down
该模块可以包含在项目中,以支持使用 Docker Compose 处理容器。
将模块依赖项添加到您的构建中,如以下 Maven 和 Gradle 清单所示:spring-boot-docker-compose
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-docker-compose</artifactId>
<optional>true</optional>
</dependency>
</dependencies>
dependencies {
developmentOnly("org.springframework.boot:spring-boot-docker-compose")
}
当此模块作为依赖项包含在内时, Spring Boot 将执行以下操作:
-
在工作目录中搜索 a 和其他常见的 compose 文件名
compose.yml
-
与发现的
docker compose up
compose.yml
-
为每个支持的容器创建服务连接 Bean
-
应用程序关闭时调用
docker compose stop
如果在启动应用程序时 Docker Compose 服务已经在运行,则 Spring Boot 将仅为每个支持的容器创建服务连接 bean。
它不会再次调用,也不会在应用程序关闭时调用。docker compose up
docker compose stop
默认情况下,重新打包的存档不包含 Spring Boot 的 Docker Compose。
如果您想使用此支持,则需要包含它。
使用 Maven 插件时,请将属性设置为 。
使用 Gradle 插件时,请配置任务的 Classpath 以包含 developmentOnly 配置。excludeDockerCompose false |
10.1. 先决条件
您需要在路径上安装 和 (或 ) CLI 应用程序。
支持的最低 Docker Compose 版本为 2.2.0。docker
docker compose
docker-compose
10.2. 服务连接
服务连接是与任何远程服务的连接。 Spring Boot 的自动配置可以使用服务连接的详细信息,并使用它们来建立与远程服务的连接。 执行此操作时,连接详细信息优先于任何与连接相关的配置属性。
当使用 Spring Boot 的 Docker Compose 支持时,将建立与容器映射的端口的服务连接。
Docker Compose 的使用方式通常是将容器内的端口映射到计算机上的临时端口。 例如,Postgres 服务器可以使用端口 5432 在容器内运行,但在本地映射到完全不同的端口。 服务连接将始终发现并使用本地映射的端口。 |
使用容器的映像名称建立服务连接。 目前支持以下服务连接:
连接详细信息 | 匹配时间 |
---|---|
|
名为 “symptoma/activemq” 的容器 |
|
名为 “cassandra” 的容器 |
|
名为 “elasticsearch” 的容器 |
|
名为“gvenzl/oracle-free”、“gvenzl/oracle-xe”、“mariadb”、“mssql/server”、“mysql”或“postgres”的容器 |
|
名为 “mongo” 的容器 |
|
名为 “neo4j” 的容器 |
|
名为 “otel/opentelemetry-collector-contrib” 的容器 |
|
名为 “otel/opentelemetry-collector-contrib” 的容器 |
|
名为 “apachepulsar/pulsar” 的容器 |
|
名为“gvenzl/oracle-free”、“gvenzl/oracle-xe”、“mariadb”、“mssql/server”、“mysql”或“postgres”的容器 |
|
名为 “rabbitmq” 的容器 |
|
名为 “redis” 的容器 |
|
名为 “openzipkin/zipkin” 的容器。 |
10.3. 自定义镜像
有时,您可能需要使用自己的映像版本来提供服务。 您可以使用任何自定义映像,只要其行为方式与标准映像相同即可。 具体而言,标准映像支持的任何环境变量也必须在自定义映像中使用。
如果您的映像使用不同的名称,则可以在文件中使用标签,以便 Spring Boot 可以提供服务连接。
使用 name 的标签提供服务名称。compose.yml
org.springframework.boot.service-connection
例如:
services:
redis:
image: 'mycompany/mycustomredis:7.0'
ports:
- '6379'
labels:
org.springframework.boot.service-connection: redis
10.4. 跳过特定容器
如果您在 中定义了一个不希望连接到应用程序的容器镜像,则可以使用标签来忽略它。
任何带有 label with 的容器都将被 Spring Boot 忽略。compose.yml
org.springframework.boot.ignore
例如:
services:
redis:
image: 'redis:7.0'
ports:
- '6379'
labels:
org.springframework.boot.ignore: true
10.5. 使用特定的 Compose 文件
如果您的 compose 文件与应用程序不在同一目录中,或者名称不同,则可以使用 in 或 来指向其他文件。
属性可以定义为精确路径或相对于应用程序的路径。spring.docker.compose.file
application.properties
application.yaml
例如:
spring.docker.compose.file=../my-compose.yml
spring:
docker:
compose:
file: "../my-compose.yml"
10.6. 等待容器就绪
由 Docker Compose 启动的容器可能需要一些时间才能完全准备就绪。
检查就绪情况的推荐方法是在文件中的服务定义下添加一个部分。healthcheck
compose.yml
由于文件中省略配置的情况并不少见,因此 Spring Boot 还直接检查服务就绪情况。
默认情况下,当可以与容器的映射端口建立 TCP/IP 连接时,容器被视为准备就绪。healthcheck
compose.yml
您可以通过在文件中添加标签来按容器禁用此功能。org.springframework.boot.readiness-check.tcp.disable
compose.yml
例如:
services:
redis:
image: 'redis:7.0'
ports:
- '6379'
labels:
org.springframework.boot.readiness-check.tcp.disable: true
您还可以更改 or 文件中的超时值:application.properties
application.yaml
spring.docker.compose.readiness.tcp.connect-timeout=10s
spring.docker.compose.readiness.tcp.read-timeout=5s
spring:
docker:
compose:
readiness:
tcp:
connect-timeout: 10s
read-timeout: 5s
总超时可使用 进行配置。spring.docker.compose.readiness.timeout
10.7. 控制 Docker Compose 生命周期
默认情况下, Spring Boot 会在应用程序启动时和关闭时调用。
如果您希望使用不同的生命周期管理,则可以使用该属性。docker compose up
docker compose stop
spring.docker.compose.lifecycle-management
支持以下值:
-
none
- 不要启动或停止 Docker Compose -
start-only
- 在应用程序启动时启动 Docker Compose 并保持运行 -
start-and-stop
- 在应用程序启动时启动 Docker Compose,在 JVM 退出时停止它
此外,还可以使用该属性来更改是否使用 or。
允许您配置 if 或 is used。spring.docker.compose.start.command
docker compose up
docker compose start
spring.docker.compose.stop.command
docker compose down
docker compose stop
以下示例显示了如何配置生命周期管理:
spring.docker.compose.lifecycle-management=start-and-stop
spring.docker.compose.start.command=start
spring.docker.compose.stop.command=down
spring.docker.compose.stop.timeout=1m
spring:
docker:
compose:
lifecycle-management: start-and-stop
start:
command: start
stop:
command: down
timeout: 1m
10.8. 激活 Docker Compose 配置文件
Docker Compose 配置文件与 Spring 配置文件类似,因为它们允许您针对特定环境调整 Docker Compose 配置。
如果要激活特定的 Docker Compose 配置文件,可以在 or 文件中使用该属性:spring.docker.compose.profiles.active
application.properties
application.yaml
spring.docker.compose.profiles.active=myprofile
spring:
docker:
compose:
profiles:
active: "myprofile"
10.9. 在测试中使用 Docker Compose
默认情况下,Spring Boot 的 Docker Compose 支持在运行测试时处于禁用状态。
要在测试中启用 Docker Compose 支持,请设置为 .spring.docker.compose.skip.in-tests
false
使用 Gradle 时,还需要将依赖项的配置从 更改为 :spring-boot-docker-compose
developmentOnly
testAndDevelopmentOnly
dependencies {
testAndDevelopmentOnly("org.springframework.boot:spring-boot-docker-compose")
}
11. Testcontainers 支持
除了使用 Testcontainers 进行集成测试外,还可以在开发时使用它们。 接下来的部分将提供有关这方面的更多详细信息。
11.1. 在开发时使用 Testcontainers
这种方法允许开发人员为应用程序所依赖的服务快速启动容器,无需手动预置数据库服务器等内容。 以这种方式使用 Testcontainers 可提供类似于 Docker Compose 的功能,但您的容器配置是 Java 而不是 YAML。
要在开发时使用 Testcontainers,您需要使用 “test” classpath 而不是 “main” 来启动应用程序。 这将允许您访问所有声明的测试依赖项,并为您提供一个自然的位置来编写测试配置。
要创建应用程序的测试可启动版本,您应该在目录中创建一个 “Application” 类。
例如,如果您的主应用程序位于 中,则应创建src/test
src/main/java/com/example/MyApplication.java
src/test/java/com/example/TestMyApplication.java
该类可以使用该方法启动真实的应用程序:TestMyApplication
SpringApplication.from(…)
public class TestMyApplication {
public static void main(String[] args) {
SpringApplication.from(MyApplication::main).run(args);
}
}
fun main(args: Array<String>) {
fromApplication<MyApplication>().run(*args)
}
您还需要定义要与应用程序一起启动的实例。
为此,您需要确保已将模块添加为依赖项。
完成后,您可以创建一个类,用于为要启动的容器声明方法。Container
spring-boot-testcontainers
test
@TestConfiguration
@Bean
你也可以用 for create bean 来注释你的方法。
有关支持的技术的详细信息,请参阅服务连接部分。@Bean
@ServiceConnection
ConnectionDetails
典型的 Testcontainers 配置如下所示:
@TestConfiguration(proxyBeanMethods = false)
public class MyContainersConfiguration {
@Bean
@ServiceConnection
public Neo4jContainer<?> neo4jContainer() {
return new Neo4jContainer<>("neo4j:5");
}
}
@TestConfiguration(proxyBeanMethods = false)
class MyContainersConfiguration {
@Bean
@ServiceConnection
fun neo4jContainer(): Neo4jContainer<*> {
return Neo4jContainer("neo4j:5")
}
}
bean 的生命周期由 Spring Boot 自动管理。
容器将自动启动和停止。Container |
您可以使用该属性来更改容器的启动方式。
默认情况下,使用启动,但您也可以选择是否希望并行启动多个容器。spring.testcontainers.beans.startup sequential parallel |
定义测试配置后,您可以使用该方法将其附加到测试Starters:with(…)
public class TestMyApplication {
public static void main(String[] args) {
SpringApplication.from(MyApplication::main).with(MyContainersConfiguration.class).run(args);
}
}
fun main(args: Array<String>) {
fromApplication<MyApplication>().with(MyContainersConfiguration::class).run(*args)
}
现在,您可以像启动任何常规 Java 方法应用程序一样启动应用程序,以启动应用程序及其需要运行的容器。TestMyApplication
main
您可以使用 Maven 目标或 Gradle 任务从命令行执行此操作。spring-boot:test-run bootTestRun |
11.1.1. 在开发时贡献动态属性
如果您想在开发时从方法中提供动态属性,可以通过注入 .
其工作方式与您可以在测试中使用的 @DynamicPropertySource
注解类似。
它允许您添加在容器启动后将变为可用的属性。Container
@Bean
DynamicPropertyRegistry
典型的配置如下所示:
@TestConfiguration(proxyBeanMethods = false)
public class MyContainersConfiguration {
@Bean
public MongoDBContainer mongoDbContainer(DynamicPropertyRegistry properties) {
MongoDBContainer container = new MongoDBContainer("mongo:5.0");
properties.add("spring.data.mongodb.host", container::getHost);
properties.add("spring.data.mongodb.port", container::getFirstMappedPort);
return container;
}
}
@TestConfiguration(proxyBeanMethods = false)
class MyContainersConfiguration {
@Bean
fun monogDbContainer(properties: DynamicPropertyRegistry): MongoDBContainer {
var container = MongoDBContainer("mongo:5.0")
properties.add("spring.data.mongodb.host", container::getHost)
properties.add("spring.data.mongodb.port", container::getFirstMappedPort)
return container
}
}
建议尽可能使用 a,但是,对于尚不支持的技术,动态属性可能是一个有用的后备。@ServiceConnection @ServiceConnection |
11.1.2. 导入 Testcontainer 声明类
使用 Testcontainers 时的一种常见模式是将实例声明为静态字段。
通常,这些字段直接在 test 类上定义。
它们也可以在父类或测试实现的接口上声明。Container
例如,以下接口声明和容器:MyContainers
mongo
neo4j
public interface MyContainers {
@Container
@ServiceConnection
MongoDBContainer mongoContainer = new MongoDBContainer("mongo:5.0");
@Container
@ServiceConnection
Neo4jContainer<?> neo4jContainer = new Neo4jContainer<>("neo4j:5");
}
如果您已经以这种方式定义了容器,或者您只是喜欢这种样式,则可以导入这些声明类,而不是将容器定义为方法。
为此,请将注释添加到您的测试配置类中:@Bean
@ImportTestcontainers
@TestConfiguration(proxyBeanMethods = false)
@ImportTestcontainers(MyContainers.class)
public class MyContainersConfiguration {
}
@TestConfiguration(proxyBeanMethods = false)
@ImportTestcontainers(MyContainers::class)
class MyContainersConfiguration
如果您不打算使用服务连接功能,但想改用 @DynamicPropertySource ,请从字段中删除注释。
您还可以将带注释的方法添加到声明类中。@ServiceConnection Container @DynamicPropertySource |
11.1.3. 在开发时将 DevTools 与 Testcontainers 一起使用
使用 devtools 时,可以使用 .
当 devtools 重新启动应用程序时,不会重新创建此类 bean。
这对于 Testcontainer bean 特别有用,因为尽管应用程序重新启动,它们仍保持其状态。@RestartScope
Container
@TestConfiguration(proxyBeanMethods = false)
public class MyContainersConfiguration {
@Bean
@RestartScope
@ServiceConnection
public MongoDBContainer mongoDbContainer() {
return new MongoDBContainer("mongo:5.0");
}
}
@TestConfiguration(proxyBeanMethods = false)
class MyContainersConfiguration {
@Bean
@RestartScope
@ServiceConnection
fun monogDbContainer(): MongoDBContainer {
return MongoDBContainer("mongo:5.0")
}
}
如果你使用的是 Gradle 并希望使用此功能,则需要将依赖项的配置从 更改为 。
使用默认范围 ,该任务不会获取代码中的更改,因为 devtools 未处于活动状态。spring-boot-devtools developmentOnly testAndDevelopmentOnly developmentOnly bootTestRun |
12. 创建您自己的自动配置
如果您在开发共享库的公司工作,或者如果您从事开源或商业库的工作,则可能需要开发自己的自动配置。 自动配置类可以捆绑在外部 jar 中,并且仍然由 Spring Boot 拾取。
Auto-configuration 可以与 “starter” 相关联,该 “starter” 提供 auto-configuration 代码以及您将与之一起使用的典型 libraries。 我们首先介绍构建自己的自动配置所需了解的内容,然后我们继续介绍创建自定义Starters所需的典型步骤。
12.1. 了解自动配置的 bean
实现自动配置的类用 .
这个 Comments 本身是 元 Comments 的,使自动配置成为标准类。
其他注释用于限制何时应应用自动配置。
通常,自动配置类使用 和 annotations。
这可确保仅在找到相关类且您尚未声明自己的类时应用自动配置。@AutoConfiguration
@Configuration
@Configuration
@Conditional
@ConditionalOnClass
@ConditionalOnMissingBean
@Configuration
你可以浏览spring-boot-autoconfigure
的源代码来查看 Spring 提供的类(参见META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports
文件)。@AutoConfiguration
12.2. 查找 Auto-configuration Candidate
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 注释上使用 、 和 属性
。
例如,如果您提供特定于 Web 的配置,则可能需要在 .before
beforeName
after
afterName
WebMvcAutoConfiguration
如果要订购某些不应直接了解彼此的自动配置,也可以使用 。
该 annotation 与常规 annotation 具有相同的语义,但为 auto-configuration 类提供了专用的顺序。@AutoConfigureOrder
@Order
与标准类一样,应用自动配置类的顺序仅影响其 bean 的定义顺序。
这些 bean 的后续创建顺序不受影响,并且由每个 bean 的依赖关系和任何关系决定。@Configuration
@DependsOn
12.3. 条件注解
您几乎总是希望在 auto-configuration 类中包含一个或多个 Comments。
注解是一个常见的示例,用于允许开发人员在对您的默认值不满意时覆盖自动配置。@Conditional
@ConditionalOnMissingBean
Spring Boot 包含许多 Comments,您可以通过 Comments 类或单个方法在自己的代码中重用这些 Comments。
这些注释包括:@Conditional
@Configuration
@Bean
12.3.1. 类条件
和 注解允许根据特定类的存在与否来包含类。
由于注释元数据是使用 ASM 解析的,因此您可以使用该属性来引用实际类,即使该类实际上可能并未出现在正在运行的应用程序类路径上。
如果您希望通过使用值来指定类名,也可以使用该属性。@ConditionalOnClass
@ConditionalOnMissingClass
@Configuration
value
name
String
此机制不适用于通常返回类型是条件目标的方法:在方法的条件应用之前,JVM 将加载类和可能处理的方法引用,如果类不存在,则这些引用将失败。@Bean
若要处理此方案,可以使用单独的类来隔离条件,如以下示例所示:@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();
}
}
}
@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 |
12.3.2. Bean 条件
和 注解允许根据特定 bean 的存在与否来包含 bean。
您可以使用该属性按类型指定 bean 或按名称指定 bean。
该属性允许您限制搜索 bean 时应考虑的层次结构。@ConditionalOnBean
@ConditionalOnMissingBean
value
name
search
ApplicationContext
当放置在方法上时,目标类型默认为方法的返回类型,如以下示例所示:@Bean
@AutoConfiguration
public class MyAutoConfiguration {
@Bean
@ConditionalOnMissingBean
public SomeService someService() {
return new SomeService();
}
}
@Configuration(proxyBeanMethods = false)
class MyAutoConfiguration {
@Bean
@ConditionalOnMissingBean
fun someService(): SomeService {
return SomeService()
}
}
在前面的示例中,如果 .someService
SomeService
ApplicationContext
您需要非常小心 bean 定义的添加顺序,因为这些条件是根据到目前为止已处理的内容进行评估的。
因此,我们建议仅在自动配置类上使用 and 注释(因为可以保证在添加任何用户定义的 bean 定义后加载这些 Comments)。@ConditionalOnBean @ConditionalOnMissingBean |
@ConditionalOnBean 并且不会阻止创建类。
在类级别使用这些条件与使用 Comments 标记每个包含的方法之间的唯一区别是,如果条件不匹配,前者会阻止将类注册为 Bean。@ConditionalOnMissingBean @Configuration @Bean @Configuration |
声明方法时,请在方法的 return 类型中提供尽可能多的类型信息。
例如,如果 Bean 的具体类实现了一个接口,则 Bean 方法的返回类型应该是具体类,而不是接口。
在使用 Bean 条件时,在方法中提供尽可能多的类型信息尤为重要,因为它们的评估只能依赖于方法签名中可用的类型信息。@Bean @Bean |
12.3.3. 属性条件
该注释允许根据 Spring Environment 属性包含配置。
使用 and 属性指定应检查的属性。
默认情况下,将匹配存在且不等于的任何属性。
您还可以使用 和 属性创建更高级的检查。@ConditionalOnProperty
prefix
name
false
havingValue
matchIfMissing
如果属性中给出了多个名称,则所有属性都必须通过测试才能匹配条件。name
12.3.4. 资源条件
该注释允许仅在存在特定资源时包含配置。
可以使用通常的 Spring 约定指定资源,如以下示例所示:。@ConditionalOnResource
file:/home/user/test.dat
12.3.5. Web 应用程序条件
和 annotations 允许根据应用程序是否为 Web 应用程序来包含配置。
基于 servlet 的 Web 应用程序是使用 Spring 、定义范围或具有 .
反应式 Web 应用程序是使用 . 或具有 .@ConditionalOnWebApplication
@ConditionalOnNotWebApplication
WebApplicationContext
session
ConfigurableWebEnvironment
ReactiveWebApplicationContext
ConfigurableReactiveWebEnvironment
和 注释允许根据应用程序是否是部署到 servlet 容器的传统 WAR 应用程序来包含配置。
对于使用嵌入式 Web 服务器运行的应用程序,此条件将不匹配。@ConditionalOnWarDeployment
@ConditionalOnNotWarDeployment
12.3.6. SPEL 表达式条件
该 Comments 允许根据 SPEL 表达式的结果包含配置。@ConditionalOnExpression
在表达式中引用 bean 将导致该 bean 在上下文刷新处理中非常早地初始化。 因此,bean 将不符合后处理条件(例如配置属性绑定),并且其状态可能不完整。 |
12.4. 测试您的 Auto-configuration
自动配置可能受许多因素影响:用户配置(定义和自定义)、条件评估(存在特定库)等。
具体来说,每个测试都应该创建一个定义明确的 Well Defined,用于表示这些自定义项的组合。 提供了实现此目的的好方法。@Bean
Environment
ApplicationContext
ApplicationContextRunner
ApplicationContextRunner 在本机映像中运行测试时不起作用。 |
ApplicationContextRunner
通常定义为 Test 类的字段,用于收集基本、通用的配置。
以下示例确保始终调用该 API:MyServiceAutoConfiguration
private final ApplicationContextRunner contextRunner = new ApplicationContextRunner()
.withConfiguration(AutoConfigurations.of(MyServiceAutoConfiguration.class));
val contextRunner = ApplicationContextRunner()
.withConfiguration(AutoConfigurations.of(MyServiceAutoConfiguration::class.java))
如果必须定义多个 auto-configurations,则无需对它们的声明进行排序,因为它们的调用顺序与运行应用程序时的顺序完全相同。 |
每个测试都可以使用运行程序来表示特定的使用案例。
例如,下面的示例调用用户配置 () 并检查自动配置是否正确退出。
调用提供了可与 一起使用的回调上下文。UserConfiguration
run
AssertJ
@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
@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")
}
}
运行程序还可用于显示 .
报表可以打印 或 级别。
以下示例显示如何使用 在 auto-configuration tests 中打印报告。ConditionEvaluationReport
INFO
DEBUG
ConditionEvaluationReportLoggingListener
class MyConditionEvaluationReportingTests {
@Test
void autoConfigTest() {
new ApplicationContextRunner()
.withInitializer(ConditionEvaluationReportLoggingListener.forLogLevel(LogLevel.INFO))
.run((context) -> {
// Test something...
});
}
}
class MyConditionEvaluationReportingTests {
@Test
fun autoConfigTest() {
ApplicationContextRunner()
.withInitializer(ConditionEvaluationReportLoggingListener.forLogLevel(LogLevel.INFO))
.run { context: AssertableApplicationContext? -> }
}
}
12.4.1. 模拟 Web 上下文
如果需要测试仅在 servlet 或反应式 Web 应用程序上下文中运行的自动配置,请分别使用 or。WebApplicationContextRunner
ReactiveWebApplicationContextRunner
12.4.2. 覆盖 Classpath
还可以测试当特定类和/或包在运行时不存在时会发生什么。
Spring Boot 附带了一个 Runner 可以轻松使用。
在以下示例中,我们断言如果不存在,则正确禁用自动配置:FilteredClassLoader
MyService
@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")
}
}
12.5. 创建您自己的 Starter
典型的 Spring Boot Starters包含用于自动配置和自定义给定技术基础设施的代码,我们称之为 “acme”。 为了使其易于扩展,可以将专用命名空间中的许多 Configuration Key 公开给环境。 最后,提供了一个 “starter” 依赖项,以帮助用户尽可能轻松地入门。
具体来说,自定义Starters可以包含以下内容:
-
包含 “acme” 的自动配置代码的模块。
autoconfigure
-
为模块提供依赖项的模块以及 “acme” 和通常有用的任何其他依赖项。 简而言之,添加 starter 应该提供开始使用该库所需的一切。
starter
autoconfigure
这种分为两个模块的做法是完全没有必要的。
如果 “acme” 有多种风格、选项或可选功能,那么最好将自动配置分开,因为您可以清楚地表达某些功能是可选的。
此外,您还可以制作一个 starter 来提供有关这些可选依赖项的意见。
与此同时,其他人可以只依赖该模块,并制作自己的不同意见的Starters。autoconfigure
如果自动配置相对简单且没有可选功能,那么在 starter 中合并两个模块绝对是一种选择。
12.5.1. 命名
您应该确保为 starter 提供适当的命名空间。
不要以 , 以 开头的模块名称,即使您使用不同的 Maven 也是如此。
将来,我们可能会为您自动配置的事物提供官方支持。spring-boot
groupId
根据经验,您应该在 starter 之后命名组合模块。
例如,假设您正在为 “acme” 创建一个Starters,并将 auto-configure 模块命名为 starter 和 starter 。
如果您只有一个模块将两者组合在一起,请将其命名为 。acme-spring-boot
acme-spring-boot-starter
acme-spring-boot-starter
12.5.2. 配置键
如果您的 starter 提供 Configuration Key,请为它们使用唯一的命名空间。
特别是,不要在 Spring Boot 使用的名称空间中包含您的键(例如、、、等)。
如果你使用相同的命名空间,我们将来可能会以破坏你的模块的方式修改这些命名空间。
根据经验,请在所有键前加上您拥有的 namespace (例如 )。server
management
spring
acme
通过为每个属性添加字段 javadoc 来确保记录配置键,如以下示例所示:
@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);
}
@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))
您应该只使用带有字段 Javadoc 的纯文本,因为它们在添加到 JSON 之前不会进行处理。@ConfigurationProperties |
以下是我们在内部遵循的一些规则,以确保描述一致:
-
请勿以 “The” 或 “A” 开头描述。
-
对于类型,请以 “Whether” 或 “Enable” 开始描述。
boolean
-
对于基于集合的类型,以 “Comma-separated list” 开始描述
-
如果默认单位与毫秒不同,请使用 instead instead and 并描述默认单位,例如“如果未指定持续时间后缀,则将使用秒”。
java.time.Duration
long
-
除非必须在运行时确定,否则不要在描述中提供默认值。
确保触发元数据生成,以便 IDE 帮助也可用于您的密钥。
您可能需要查看生成的元数据 () 以确保您的密钥已正确记录。
在兼容的 IDE 中使用您自己的 starter 也是验证元数据质量的好主意。META-INF/spring-configuration-metadata.json
12.5.3. “autoconfigure” 模块
该模块包含开始使用该库所需的一切。
它还可能包含配置键定义(例如 )和可用于进一步自定义组件初始化方式的任何回调接口。autoconfigure
@ConfigurationProperties
您应该将库的依赖项标记为可选,以便可以更轻松地将模块包含在项目中。
如果这样做,则不会提供库,并且默认情况下, Spring Boot 会退缩。autoconfigure |
Spring Boot 使用 Comments 处理器在元数据文件 () 中收集自动配置的条件。
如果存在该文件,则使用它来紧急过滤不匹配的自动配置,这将缩短启动时间。META-INF/spring-autoconfigure-metadata.properties
使用 Maven 构建时,建议在包含 auto-configurations 的模块中添加以下依赖项:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-autoconfigure-processor</artifactId>
<optional>true</optional>
</dependency>
如果您直接在应用程序中定义了 auto-configurations,请确保配置 以防止目标将依赖项添加到 uber jar 中:spring-boot-maven-plugin
repackage
<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"
}
12.5.4. Starter 模块
Starters真的是一个空罐子。 它的唯一目的是提供使用库所需的依赖项。 您可以将其视为对开始所需内容的固执己见的看法。
不要对添加起始项的项目做出假设。 如果要自动配置的库通常需要其他启动程序,请同时提及它们。 如果可选依赖项的数量很高,则可能很难提供一组适当的默认依赖项,因为您应该避免包含对于库的典型使用不必要的依赖项。 换句话说,您不应包含可选依赖项。
无论哪种方式,你的 starter 都必须直接或间接引用核心 Spring Boot starter () (如果你的 starter 依赖于另一个 starter,则无需添加它)。
如果项目是仅使用您的自定义Starters创建的,则 Spring Boot 的核心功能将因核心Starters的存在而得到尊重。spring-boot-starter |
13. Kotlin 支持
Spring Boot 通过利用其他 Spring 项目(如 Spring Framework、Spring Data 和 Reactor)中的支持来提供 Kotlin 支持。 有关更多信息,请参阅 Spring Framework Kotlin 支持文档。
开始使用 Spring Boot 和 Kotlin 的最简单方法是遵循此综合教程。
您可以使用 start.spring.io 创建新的 Kotlin 项目。
如果您需要支持,请随时加入 Kotlin Slack 的 #spring 频道,或者在 Stack Overflow 上使用 和 标签提问。spring
kotlin
13.1. 要求
Spring Boot 至少需要 Kotlin 1.7.x,并通过依赖项管理管理合适的 Kotlin 版本。
要使用 Kotlin,并且必须存在于 Classpath 中。
变体 和 也可以使用。org.jetbrains.kotlin:kotlin-stdlib
org.jetbrains.kotlin:kotlin-reflect
kotlin-stdlib
kotlin-stdlib-jdk7
kotlin-stdlib-jdk8
由于 Kotlin 类默认是 final,因此您可能希望配置 kotlin-spring 插件,以便自动打开 Spring 注释的类,以便可以代理它们。
在 Kotlin 中序列化/反序列化 JSON 数据需要 Jackson 的 Kotlin 模块。 在 Classpath 上找到它时,它会自动注册。 如果存在 Jackson 和 Kotlin,但不存在 Jackson Kotlin 模块,则会记录一条警告消息。
如果在 start.spring.io 上引导 Kotlin 项目,则默认提供这些依赖项和插件。 |
13.2. 空安全
Kotlin 的主要功能之一是 null 安全。
它在编译时处理值,而不是将问题推迟到运行时并遇到 .
这有助于消除常见的错误来源,而无需支付像 .
Kotlin 还允许使用具有可为 null 值的函数结构,如本 Kotlin 中的 null 安全性综合指南中所述。null
NullPointerException
Optional
尽管 Java 不允许在其类型系统中表达 null 安全性,但 Spring Framework、Spring Data 和 Reactor 现在通过工具友好的注释提供其 API 的 null 安全性。 默认情况下,Kotlin 中使用的 Java API 中的类型被识别为放宽了 null 检查的平台类型。Kotlin 对 JSR 305 注释的支持与可为 null 性注释相结合,为 Kotlin 中的相关 Spring API 提供了空安全性。
可以通过添加编译器标志和以下选项来配置 JSR 305 检查:.
默认行为与 相同。
在从 Spring API 推断的 Kotlin 类型中,需要考虑 null-safety 该值,但应在知道 Spring API 可空性声明甚至在次要版本之间演变,并且将来可能会添加更多检查的情况下使用该值。-Xjsr305
-Xjsr305={strict|warn|ignore}
-Xjsr305=warn
strict
13.3. Kotlin API
13.3.1. runApplication
Spring Boot 提供了一种惯用的方式来运行应用程序,如以下示例所示:runApplication<MyApplication>(*args)
@SpringBootApplication
class MyApplication
fun main(args: Array<String>) {
runApplication<MyApplication>(*args)
}
这是 的直接替代品。
它还允许自定义应用程序,如以下示例所示:SpringApplication.run(MyApplication::class.java, *args)
runApplication<MyApplication>(*args) {
setBannerMode(OFF)
}
13.3.2. 扩展
Kotlin 扩展提供了使用其他功能扩展现有类的功能。 Spring Boot Kotlin API 利用这些扩展为现有 API 添加新的 Kotlin 特定便利。
TestRestTemplate
提供了扩展,类似于 Spring Framework 为 Spring Framework 提供的扩展。
此外,这些扩展还可以利用 Kotlin 具体化类型参数。RestOperations
13.4. 依赖项管理
为了避免在 Classpath 上混合不同版本的 Kotlin 依赖项, Spring Boot 导入了 Kotlin BOM。
使用 Maven,可以通过设置属性来自定义 Kotlin 版本,并为 .
使用 Gradle 时,Spring Boot 插件会自动与 Kotlin 插件的版本保持一致。kotlin.version
kotlin-maven-plugin
kotlin.version
Spring Boot 还通过导入 Kotlin Coroutines BOM 来管理 Coroutines 依赖项的版本。
可以通过设置属性来自定义版本。kotlin-coroutines.version
org.jetbrains.kotlinx:kotlinx-coroutines-reactor 如果引导 Kotlin 项目时至少有一个对 start.spring.io 的反应式依赖项,则默认提供依赖项。 |
13.5. @ConfigurationProperties
@ConfigurationProperties
与构造函数结合使用时,绑定支持具有不可变属性的数据类,如以下示例所示:val
@ConfigurationProperties("example.kotlin")
data class KotlinExampleProperties(
val name: String,
val description: String,
val myService: MyService) {
data class MyService(
val apiToken: String,
val uri: URI
)
}
由于它们与 Java 的互操作性受到限制,因此对值类的支持受到限制。 特别是,依赖 value 类的默认值不适用于 configuration property binding。 在这种情况下,应改用数据类。
要使用 annotation 处理器生成您自己的元数据,应该为 kapt 配置依赖项。
请注意,由于 kapt 提供的模型的限制,某些功能(例如检测默认值或已弃用的项目)无法正常工作。spring-boot-configuration-processor |
13.6. 测试
虽然可以使用 JUnit 4 来测试 Kotlin 代码,但 JUnit 5 是默认提供的,建议使用 JUnit 5。
JUnit 5 允许测试类实例化一次,并重复用于该类的所有测试。
这使得在非静态方法上使用 and 注解成为可能,这非常适合 Kotlin。@BeforeAll
@AfterAll
要模拟 Kotlin 类,建议使用 MockK。
如果你需要 Mockito 特定的 @MockBean
和 @SpyBean
注释的等效项,你可以使用 SpringMockK,它提供类似的 COMMENTS 。MockK
@MockkBean
@SpykBean
13.7. 资源
13.7.1. 延伸阅读
-
Kotlin Slack(具有专用的 #spring 频道)
13.7.2. 示例
-
spring-boot-kotlin-demo:常规 Spring Boot + Spring Data JPA 项目
-
mixit:Spring Boot 2 + WebFlux + 反应式 Spring Data MongoDB
-
spring-kotlin-fullstack:WebFlux Kotlin 全栈示例,前端使用 Kotlin2js 而不是 JavaScript 或 TypeScript
-
spring-petclinic-kotlin:Spring PetClinic 示例应用程序的 Kotlin 版本
-
spring-kotlin-deepdive:从 Boot 1.0 + Java 到 Boot 2.0 + Kotlin 的分步迁移
-
spring-boot-coroutines-demo:协程示例项目
14. SSL认证
Spring Boot 提供了配置 SSL 信任材料的能力,该材料可以应用于多种类型的连接,以支持安全通信。
带有前缀的配置属性可用于指定命名的信任材料和相关信息集。spring.ssl.bundle
14.1. 使用 Java KeyStore 文件配置 SSL
带有前缀的配置属性可用于配置使用 Java 实用程序创建并以 JKS 或 PKCS12 格式存储在 Java KeyStore 文件中的信任材料包。
每个捆绑包都有一个用户提供的名称,可用于引用捆绑包。spring.ssl.bundle.jks
keytool
当用于保护嵌入式 Web 服务器时,通常配置包含证书和私钥的 Java KeyStore,如以下示例所示:keystore
spring.ssl.bundle.jks.mybundle.key.alias=application
spring.ssl.bundle.jks.mybundle.keystore.location=classpath:application.p12
spring.ssl.bundle.jks.mybundle.keystore.password=secret
spring.ssl.bundle.jks.mybundle.keystore.type=PKCS12
spring:
ssl:
bundle:
jks:
mybundle:
key:
alias: "application"
keystore:
location: "classpath:application.p12"
password: "secret"
type: "PKCS12"
当用于保护客户端连接时,通常配置包含服务器证书的 Java KeyStore,如以下示例所示:truststore
spring.ssl.bundle.jks.mybundle.truststore.location=classpath:server.p12
spring.ssl.bundle.jks.mybundle.truststore.password=secret
spring:
ssl:
bundle:
jks:
mybundle:
truststore:
location: "classpath:server.p12"
password: "secret"
有关完整的受支持属性集,请参阅 JksSslBundleProperties。
14.2. 使用 PEM 编码的证书配置 SSL
带有前缀的配置属性可用于以 PEM 编码文本的形式配置信任材料包。
每个捆绑包都有一个用户提供的名称,可用于引用捆绑包。spring.ssl.bundle.pem
当用于保护嵌入式 Web 服务器时,通常配置有证书和私钥,如以下示例所示:keystore
spring.ssl.bundle.pem.mybundle.keystore.certificate=classpath:application.crt
spring.ssl.bundle.pem.mybundle.keystore.private-key=classpath:application.key
spring:
ssl:
bundle:
pem:
mybundle:
keystore:
certificate: "classpath:application.crt"
private-key: "classpath:application.key"
当用于保护客户端连接时,通常使用服务器证书配置 a,如以下示例所示:truststore
spring.ssl.bundle.pem.mybundle.truststore.certificate=classpath:server.crt
spring:
ssl:
bundle:
pem:
mybundle:
truststore:
certificate: "classpath:server.crt"
PEM 内容可以直接用于 和 属性。
如果属性值包含 和 标记,则它们将被视为 PEM 内容,而不是资源位置。 以下示例显示了如何定义信任库证书: 性能
Yaml
|
有关完整的受支持属性集,请参阅 PemSslBundleProperties。
14.3. 应用 SSL 捆绑包
使用属性配置后,可以在 Spring Boot 自动配置的各种类型的连接的配置属性中按名称引用 SSL 捆绑包。 有关详细信息,请参阅有关嵌入式 Web 服务器、数据技术和 REST 客户端的部分。
14.4. 使用 SSL 捆绑包
Spring Boot 自动配置一个类型的 bean,该 bean 提供对使用属性配置的每个命名捆绑包的访问。SslBundles
spring.ssl.bundle
可以从自动配置的 bean 中检索 an,并用于创建用于在 Client 端库中配置 SSL 连接的对象。
提供了获取这些 SSL 对象的分层方法:SslBundle
SslBundles
SslBundle
-
getStores()
提供对密钥存储和信任存储实例以及任何必需的密钥存储密码的访问。java.security.KeyStore
-
getManagers()
提供对 AND 实例以及它们创建的 AND 数组的访问。java.net.ssl.KeyManagerFactory
java.net.ssl.TrustManagerFactory
java.net.ssl.KeyManager
java.net.ssl.TrustManager
-
createSslContext()
提供了一种获取新实例的便捷方法。java.net.ssl.SSLContext
此外,它还提供有关正在使用的密钥、要使用的协议以及应应用于 SSL 引擎的任何选项的详细信息。SslBundle
以下示例显示了如何检索 an 并使用它来创建一个 :SslBundle
SSLContext
@Component
public class MyComponent {
public MyComponent(SslBundles sslBundles) {
SslBundle sslBundle = sslBundles.getBundle("mybundle");
SSLContext sslContext = sslBundle.createSslContext();
// do something with the created sslContext
}
}
@Component
class MyComponent(sslBundles: SslBundles) {
init {
val sslBundle = sslBundles.getBundle("mybundle")
val sslContext = sslBundle.createSslContext()
// do something with the created sslContext
}
}
14.5. 重新加载 SSL 捆绑包
当密钥材料发生更改时,可以重新加载 SSL 捆绑包。 使用捆绑包的组件必须与可重新加载的 SSL 捆绑包兼容。 目前兼容以下组件:
-
Tomcat Web 服务器
-
Netty Web 服务器
要启用重新加载,您需要通过配置属性选择加入,如以下示例所示:
spring.ssl.bundle.pem.mybundle.reload-on-update=true
spring.ssl.bundle.pem.mybundle.keystore.certificate=file:/some/directory/application.crt
spring.ssl.bundle.pem.mybundle.keystore.private-key=file:/some/directory/application.key
spring:
ssl:
bundle:
pem:
mybundle:
reload-on-update: true
keystore:
certificate: "file:/some/directory/application.crt"
private-key: "file:/some/directory/application.key"
然后,文件观察器将监视文件,如果文件发生更改,将重新加载 SSL 捆绑包。 这反过来会触发消费组件中的重新加载,例如 Tomcat 轮换启用 SSL 的连接器中的证书。
您可以使用属性配置文件观察程序的静默期(以确保没有更多更改)。spring.ssl.bundle.watch.file.quiet-period
15. 下一步要读什么
如果您想了解有关本节中讨论的任何类的更多信息,请参阅 Spring Boot API 文档,或者您可以直接浏览源代码。 如果您有具体问题,请参阅操作方法部分。
如果您对 Spring Boot 的核心功能感到满意,则可以继续阅读有关生产就绪功能的信息。