spring_reference/How-to_ guides.md

12 KiB
Raw Blame History

How-to指南

本章节将回答一些常见的"我该怎么做"类型的问题这些问题在我们使用Spring Boot时经常遇到。这绝不是一个详尽的列表但它覆盖了很多方面。

如果遇到一个特殊的我们没有覆盖的问题,你可能想去查看stackoverflow.com,看是否有人已经给出了答案;这也是一个很好的提新问题的地方(请使用spring-boot标签)。

我们也乐意扩展本章节;如果想添加一个'how-to',你可以给我们发一个pull请求

Spring Boot应用

  • 解决自动配置问题

Spring Boot自动配置总是尝试尽最大努力去做正确的事但有时候会失败并且很难说出失败原因。

在每个Spring Boot ApplicationContext中都存在一个相当有用的ConditionEvaluationReport。如果开启DEBUG日志输出,你将会看到它。如果你使用spring-boot-actuator,则会有一个autoconfig的端点它将以JSON形式渲染该报告。可以使用它调试应用程序并能查看Spring Boot运行时都添加了哪些特性及哪些没添加

通过查看源码和javadoc可以获取更多问题的答案。以下是一些经验

  • 查找名为*AutoConfiguration的类并阅读源码,特别是@Conditional*注解,这可以帮你找出它们启用哪些特性及何时启用。 将--debug添加到命令行或添加系统属性-Ddebug可以在控制台查看日志该日志会记录你的应用中所有自动配置的决策。在一个运行的Actuator app中通过查看autoconfig端点(/autoconfig或等效的JMX可以获取相同信息。
  • 查找是@ConfigurationProperties的类(比如ServerProperties)并看下有哪些可用的外部配置选项。@ConfigurationProperties类有一个用于充当外部配置前缀的name属性,因此ServerProperties的值为prefix="server",它的配置属性有server.portserver.address等。在运行的Actuator应用中可以查看configprops端点。
  • 查看使用RelaxedEnvironment明确地将配置从Environment暴露出去。它经常会使用一个前缀。
  • 查看@Value注解,它直接绑定到Environment。相比RelaxedEnvironment这种方式稍微缺乏灵活性但它也允许松散的绑定特别是OS环境变量所以CAPITALS_AND_UNDERSCORESperiod.separated的同义词)。
  • 查看@ConditionalOnExpression注解它根据SpEL表达式的结果来开启或关闭特性通常使用解析自Environment的占位符进行计算。
  • 启动前自定义Environment或ApplicationContext

每个SpringApplication都有ApplicationListenersApplicationContextInitializers用于自定义上下文context或环境(environment)。Spring Boot从META-INF/spring.factories下加载很多这样的内部使用的自定义。有很多方法可以注册其他的自定义:

  1. 以编程方式为每个应用注册自定义通过在SpringApplication运行前调用它的addListenersaddInitializers方法来实现。
  2. 以声明方式为每个应用注册自定义,通过设置context.initializer.classescontext.listener.classes来实现。
  3. 以声明方式为所有应用注册自定义,通过添加一个META-INF/spring.factories并打包成一个jar文件该应用将它作为一个库来实现。

SpringApplication会给监听器(即使是在上下文被创建之前就存在的)发送一些特定的ApplicationEvents,然后也会注册监听ApplicationContext发布的事件的监听器。查看Spring Boot特性章节中的Section 22.4, “Application events and listeners” 可以获取一个完整列表。

  • 构建ApplicationContext层次结构添加父或根上下文

你可以使用ApplicationBuilder类创建父/根ApplicationContext层次结构。查看'Spring Boot特性'章节的Section 22.3, “Fluent builder API” 获取更多信息。

  • 创建一个非webnon-web应用

不是所有的Spring应用都必须是web应用或web服务。如果你想在main方法中执行一些代码但需要启动一个Spring应用去设置需要的底层设施那使用Spring Boot的SpringApplication特性可以很容易实现。SpringApplication会根据它是否需要一个web应用来改变它的ApplicationContext类。首先你需要做的是去掉servlet API依赖如果不能这样做比如基于相同的代码运行两个应用那你可以明确地调用SpringApplication.setWebEnvironment(false)或设置applicationContextClass属性通过Java API或使用外部配置。你想运行的作为业务逻辑的应用代码可以实现为一个CommandLineRunner,并将上下文降级为一个@Bean定义。

属性&配置

  • 外部化SpringApplication配置

SpringApplication已经被属性化主要是setters所以你可以在创建应用时使用它的Java API修改它的行为。或者你可以使用properties文件中的spring.main.*来外部化(在应用代码外配置)这些配置。比如,在application.properties中可能会有以下内容:

spring.main.web_environment=false
spring.main.show_banner=false

然后Spring Boot在启动时将不会显示banner并且该应用也不是一个web应用。

  • 改变应用程序外部配置文件的位置

默认情况下来自不同源的属性以一个定义好的顺序添加到Spring的Environment中(查看'Sprin Boot特性'章节的Chapter 23, Externalized Configuration获取精确的顺序)。

为应用程序源添加@PropertySource注解是一种很好的添加和修改源顺序的方法。传递给SpringApplication静态便利设施convenience方法的类和使用setSources()添加的类都会被检查,以查看它们是否有@PropertySources,如果有,这些属性会被尽可能早的添加到Environment里,以确保ApplicationContext生命周期的所有阶段都能使用。以这种方式添加的属性优先于任何使用默认位置添加的属性,但低于系统属性,环境变量或命令行参数。

你也可以提供系统属性(或环境变量)来改变该行为:

  1. spring.config.nameSPRING_CONFIG_NAME)是根文件名,默认为application
  2. spring.config.locationSPRING_CONFIG_LOCATION是要加载的文件例如一个classpath资源或一个URL。Spring Boot为该文档设置一个单独的Environment属性,它可以被系统属性,环境变量或命令行参数覆盖。

不管你在environment设置什么Spring Boot都将加载上面讨论过的application.properties。如果使用YAML那具有'.yml'扩展的文件默认也会被添加到该列表。

详情参考ConfigFileApplicationListener

  • 使用'short'命令行参数

有些人喜欢使用(例如)--port=9000代替--server.port=9000来设置命令行配置属性。你可以通过在application.properties中使用占位符来启用该功能比如

server.port=${port:8080}

:如果你继承自spring-boot-starter-parent POM为了防止和Spring-style的占位符产生冲突maven-resources-plugins默认的过滤令牌filter token已经从${*}变为@(即@maven.token@代替了${maven.token}。如果已经直接启用maven对application.properties的过滤你可能也想使用其他的分隔符替换默认的过滤令牌。

在这种特殊的情况下端口绑定能够在一个PaaS环境下工作比如Heroku和Cloud Foundry因为在这两个平台中PORT环境变量是自动设置的并且Spring能够绑定Environment属性的大写同义词。

  • 使用YAML配置外部属性

YAML是JSON的一个超集可以非常方便的将外部配置以层次结构形式存储起来。比如

spring:
    application:
        name: cruncher
    datasource:
        driverClassName: com.mysql.jdbc.Driver
        url: jdbc:mysql://localhost/test
server:
    port: 9000

创建一个application.yml文件将它放到classpath的根目录下并添加snakeyaml依赖Maven坐标为org.yaml:snakeyaml,如果你使用spring-boot-starter那就已经被包含了。一个YAML文件会被解析为一个Java Map<String,Object>和一个JSON对象类似Spring Boot会平伸该map这样它就只有1级深度并且有period-separated的keys跟人们在Java中经常使用的Properties文件非常类似。 上面的YAML示例对应于下面的application.properties文件

spring.application.name=cruncher
spring.datasource.driverClassName=com.mysql.jdbc.Driver
spring.datasource.url=jdbc:mysql://localhost/test
server.port=9000

查看'Spring Boot特性'章节的Section 23.6, “Using YAML instead of Properties”可以获取更多关于YAML的信息。

  • 设置生效的Spring profiles

Spring Environment有一个API可以设置生效的profiles但通常你会设置一个系统profilespring.profiles.active或一个OS环境变量SPRING_PROFILES_ACTIVE)。比如,使用一个-D参数启动应用程序记着把它放到main类或jar文件之前

$ java -jar -Dspring.profiles.active=production demo-0.0.1-SNAPSHOT.jar

在Spring Boot中你也可以在application.properties里设置生效的profile例如

spring.profiles.active=production

通过这种方式设置的值会被系统属性或环境变量替换,但不会被SpringApplicationBuilder.profiles()方法替换。因此后面的Java API可用来在不改变默认设置的情况下增加profiles。

想要获取更多信息可查看'Spring Boot特性'章节的Chapter 24, Profiles

  • 根据环境改变配置

一个YAML文件实际上是一系列以---线分割的文档每个文档都被单独解析为一个平坦的flattenedmap。

如果一个YAML文档包含一个spring.profiles关键字那profiles的值以逗号分割的profiles列表将被传入Spring的Environment.acceptsProfiles()方法并且如果这些profiles的任何一个被激活对应的文档被包含到最终的合并中否则不会

示例:

server:
    port: 9000
---

spring:
    profiles: development
server:
    port: 9001

---

spring:
    profiles: production
server:
    port: 0

在这个示例中默认的端口是9000但如果Spring profile 'development'生效则该端口是9001如果'production'生效则它是0。

YAML文档以它们遇到的顺序合并所以后面的值会覆盖前面的值

想要使用profiles文件完成同样的操作你可以使用application-${profile}.properties指定特殊的profile相关的值。

  • 发现外部属性的内置选项

Spring Boot在运行时将来自application.properties或.yml的外部属性绑定进一个应用中。在一个地方不可能存在详尽的所有支持属性的列表技术上也是不可能的因为你的classpath下的其他jar文件也能够贡献。

每个运行中且有Actuator特性的应用都会有一个configprops端点,它能够展示所有边界和可通过@ConfigurationProperties绑定的属性。

附录中包含一个application.properties示例它列举了Spring Boot支持的大多数常用属性。获取权威列表可搜索@ConfigurationProperties@Value的源码,还有不经常使用的RelaxedEnvironment

内嵌的servlet容器