本章节将回答一些常见的”我该怎么做”类型的问题,这些问题在我们使用Spring Boot时经常遇到。这绝不是一个详尽的列表,但它覆盖了很多方面。
如果遇到一个特殊的我们没有覆盖的问题,你可能想去查看stackoverflow.com,看是否有人已经给出了答案;这也是一个很好的提新问题的地方(请使用spring-boot
标签)。
我们也乐意扩展本章节;如果想添加一个’how-to’,你可以给我们发一个pull请求。
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.port
,server.address
等。在运行的Actuator应用中可以查看configprops
端点。RelaxedEnvironment
明确地将配置从Environment
暴露出去。它经常会使用一个前缀。@Value
注解,它直接绑定到Environment
。相比RelaxedEnvironment
,这种方式稍微缺乏灵活性,但它也允许松散的绑定,特别是OS环境变量(所以CAPITALS_AND_UNDERSCORES
是period.separated
的同义词)。查看@ConditionalOnExpression
注解,它根据SpEL表达式的结果来开启或关闭特性,通常使用解析自Environment
的占位符进行计算。
启动前自定义Environment或ApplicationContext
每个SpringApplication
都有ApplicationListeners
和ApplicationContextInitializers
,用于自定义上下文(context)或环境(environment)。Spring Boot从META-INF/spring.factories
下加载很多这样的内部使用的自定义。有很多方法可以注册其他的自定义:
addListeners
和addInitializers
方法来实现。context.initializer.classes
或context.listener.classes
来实现。META-INF/spring.factories
并打包成一个jar文件(该应用将它作为一个库)来实现。SpringApplication
会给监听器(即使是在上下文被创建之前就存在的)发送一些特定的ApplicationEvents
,然后也会注册监听ApplicationContext
发布的事件的监听器。查看Spring Boot特性章节中的Section 22.4, “Application events and listeners” 可以获取一个完整列表。
你可以使用ApplicationBuilder
类创建父/根ApplicationContext
层次结构。查看’Spring Boot特性’章节的Section 22.3, “Fluent builder API” 获取更多信息。
不是所有的Spring应用都必须是web应用(或web服务)。如果你想在main方法中执行一些代码,但需要启动一个Spring应用去设置需要的底层设施,那使用Spring Boot的SpringApplication
特性可以很容易实现。SpringApplication
会根据它是否需要一个web应用来改变它的ApplicationContext
类。首先你需要做的是去掉servlet API依赖,如果不能这样做(比如,基于相同的代码运行两个应用),那你可以明确地调用SpringApplication.setWebEnvironment(false)
或设置applicationContextClass
属性(通过Java API或使用外部配置)。你想运行的,作为业务逻辑的应用代码可以实现为一个CommandLineRunner
,并将上下文降级为一个@Bean
定义。
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
生命周期的所有阶段都能使用。以这种方式添加的属性优先于任何使用默认位置添加的属性,但低于系统属性,环境变量或命令行参数。
你也可以提供系统属性(或环境变量)来改变该行为:
spring.config.name
(SPRING_CONFIG_NAME
)是根文件名,默认为application
。spring.config.location
(SPRING_CONFIG_LOCATION
)是要加载的文件(例如,一个classpath资源或一个URL)。Spring Boot为该文档设置一个单独的Environment
属性,它可以被系统属性,环境变量或命令行参数覆盖。不管你在environment设置什么,Spring Boot都将加载上面讨论过的application.properties
。如果使用YAML,那具有’.yml’扩展的文件默认也会被添加到该列表。
详情参考ConfigFileApplicationListener
有些人喜欢使用(例如)--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是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
(和一个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 Environment
有一个API可以设置生效的profiles,但通常你会设置一个系统profile(spring.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文件实际上是一系列以---
线分割的文档,每个文档都被单独解析为一个平坦的(flattened)map。
如果一个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规范支持的Servlet,Filter,ServletContextListener和其他监听器可以作为@Bean
定义添加到你的应用中。需要格外小心的是,它们不会引起太多的其他beans的热初始化,因为在应用生命周期的早期它们已经被安装到容器里了(比如,让它们依赖你的DataSource或JPA配置就不是一个好主意)。你可以通过延迟初始化它们到第一次使用而不是初始化时来突破该限制。
在Filters和Servlets的情况下,你也可以通过添加一个FilterRegistrationBean
或ServletRegistrationBean
代替或以及底层的组件来添加映射(mappings)和初始化参数。
正如以上讨论的任何Servlet或Filter beans将被自动注册到servlet容器中。为了禁止注册一个特殊的Filter或Servlet bean,可以为它创建一个注册bean,然后禁用该bean。例如:
@Bean
public FilterRegistrationBean registration(MyFilter filter) {
FilterRegistrationBean registration = new FilterRegistrationBean(filter);
registration.setEnabled(false);
return registration;
}
在一个单独的应用中,主HTTP端口默认为8080,但可以使用server.port
设置(比如,在application.properties中或作为一个系统属性)。由于Environment
值的宽松绑定,你也可以使用SERVER_PORT
(比如,作为一个OS环境变)。
为了完全关闭HTTP端点,但仍创建一个WebApplicationContext,你可以设置server.port=-1
(测试时可能有用)。
想获取更多详情可查看’Spring Boot特性’章节的Section 26.3.3, “Customizing embedded servlet containers”,或ServerProperties源码。
想扫描一个未使用的端口(为了防止冲突使用OS本地端口)可以使用server.port=0
。
你可以通过日志输出或它的EmbeddedServletContainer的EmbeddedWebApplicationContext获取服务器正在运行的端口。获取和确认服务器已经初始化的最好方式是添加一个ApplicationListener
类型的@Bean
,然后当事件发布时将容器pull出来。
使用@WebIntegrationTests
的一个有用实践是设置server.port=0
,然后使用@Value
注入实际的(’local’)端口。例如:
@RunWith(SpringJUnit4ClassRunner.class)
@SpringApplicationConfiguration(classes = SampleDataJpaApplication.class)
@WebIntegrationTest("server.port:0")
public class CityRepositoryIntegrationTests {
@Autowired
EmbeddedWebApplicationContext server;
@Value("${local.server.port}")
int port;
// ...
}
SSL能够以声明方式进行配置,一般通过在application.properties或application.yml设置各种各样的server.ssl.*
属性。例如:
server.port = 8443
server.ssl.key-store = classpath:keystore.jks
server.ssl.key-store-password = secret
server.ssl.key-password = another-secret
获取所有支持的配置详情可查看Ssl。
注:Tomcat要求key存储(如果你正在使用一个可信存储)能够直接在文件系统上访问,即它不能从一个jar文件内读取。Jetty和Undertow没有该限制。
使用类似于以上示例的配置意味着该应用将不在支持端口为8080的普通HTTP连接。Spring Boot不支持通过application.properties同时配置HTTP连接器和HTTPS连接器。如果你两个都想要,那就需要以编程的方式配置它们中的一个。推荐使用application.properties配置HTTPS,因为HTTP连接器是两个中最容易以编程方式进行配置的。获取示例可查看spring-boot-sample-tomcat-multi-connectors示例项目。
通常你可以遵循Section 63.7, “Discover built-in options for external properties”关于@ConfigurationProperties
(这里主要的是ServerProperties
)的建议,但也看下EmbeddedServletContainerCustomizer
和各种你可以添加的Tomcat-specific的*Customizers
。
Tomcat APIs相当丰富,一旦获取到TomcatEmbeddedServletContainerFactory
,你就能够以多种方式修改它。或核心选择是添加你自己的TomcatEmbeddedServletContainerFactory
。
你可以将一个org.apache.catalina.connector.Connector
添加到TomcatEmbeddedServletContainerFactory
,这就能够允许多连接器,比如HTTP和HTTPS连接器:
@Bean
public EmbeddedServletContainerFactory servletContainer() {
TomcatEmbeddedServletContainerFactory tomcat = new TomcatEmbeddedServletContainerFactory();
tomcat.addAdditionalTomcatConnectors(createSslConnector());
return tomcat;
}
private Connector createSslConnector() {
Connector connector = new Connector("org.apache.coyote.http11.Http11NioProtocol");
Http11NioProtocol protocol = (Http11NioProtocol) connector.getProtocolHandler();
try {
File keystore = new ClassPathResource("keystore").getFile();
File truststore = new ClassPathResource("keystore").getFile();
connector.setScheme("https");
connector.setSecure(true);
connector.setPort(8443);
protocol.setSSLEnabled(true);
protocol.setKeystoreFile(keystore.getAbsolutePath());
protocol.setKeystorePass("changeit");
protocol.setTruststoreFile(truststore.getAbsolutePath());
protocol.setTruststorePass("changeit");
protocol.setKeyAlias("apitester");
return connector;
}
catch (IOException ex) {
throw new IllegalStateException("can't access keystore: [" + "keystore"
+ "] or truststore: [" + "keystore" + "]", ex);
}
}
Spring Boot将自动配置Tomcat的RemoteIpValve
,如果你启用它的话。这允许你透明地使用标准的x-forwarded-for
和x-forwarded-proto
头,很多前端代理服务器都会添加这些头信息(headers)。通过将这些属性中的一个或全部设置为非空的内容来开启该功能(它们是大多数代理约定的值,如果你只设置其中的一个,则另一个也会被自动设置)。
server.tomcat.remote_ip_header=x-forwarded-for
server.tomcat.protocol_header=x-forwarded-proto
如果你的代理使用不同的头部(headers),你可以通过向application.properties添加一些条目来自定义该值的配置,比如:
server.tomcat.remote_ip_header=x-your-remote-ip-header
server.tomcat.protocol_header=x-your-protocol-header
该值也可以配置为一个默认的,能够匹配信任的内部代理的正则表达式。默认情况下,受信任的IP包括 10/8, 192.168/16, 169.254/16 和 127/8。可以通过向application.properties添加一个条目来自定义该值的配置,比如:
server.tomcat.internal_proxies=192\\.168\\.\\d{
1,3}\\.\\d{
1,3}
注:只有在你使用一个properties文件作为配置的时候才需要双反斜杠。如果你使用YAML,单个反斜杠就足够了,192\.168\.\d{1,3}\.\d{1,3}
和上面的等价。
另外,通过在一个TomcatEmbeddedServletContainerFactory
bean中配置和添加RemoteIpValve
,你就可以完全控制它的设置了。
Spring Boot starters(特别是spring-boot-starter-web)默认都是使用Tomcat作为内嵌容器的。你需要排除那些Tomcat的依赖并包含Jetty的依赖。为了让这种处理尽可能简单,Spring Boot将Tomcat和Jetty的依赖捆绑在一起,然后提供单独的starters。
Maven示例:
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-webartifactId>
<exclusions>
<exclusion>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-tomcatartifactId>
exclusion>
exclusions>
dependency>
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-jettyartifactId>
dependency>
Gradle示例:
configurations {
compile.exclude module: "spring-boot-starter-tomcat"
}
dependencies {
compile("org.springframework.boot:spring-boot-starter-web:1.3.0.BUILD-SNAPSHOT")
compile("org.springframework.boot:spring-boot-starter-jetty:1.3.0.BUILD-SNAPSHOT")
// ...
}
通常你可以遵循Section 63.7, “Discover built-in options for external properties”关于@ConfigurationProperties
(此处主要是ServerProperties)的建议,但也要看下EmbeddedServletContainerCustomizer
。Jetty API相当丰富,一旦获取到JettyEmbeddedServletContainerFactory
,你就可以使用很多方式修改它。或更彻底地就是添加你自己的JettyEmbeddedServletContainerFactory
。
使用Undertow替代Tomcat和使用Jetty替代Tomcat非常类似。你需要排除Tomat依赖,并包含Undertow starter。
Maven示例:
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-webartifactId>
<exclusions>
<exclusion>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-tomcatartifactId>
exclusion>
exclusions>
dependency>
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-undertowartifactId>
dependency>
Gradle示例:
configurations {
compile.exclude module: "spring-boot-starter-tomcat"
}
dependencies {
compile 'org.springframework.boot:spring-boot-starter-web:1.3.0.BUILD-SNAPSHOT")
compile 'org.springframework.boot:spring-boot-starter-undertow:1.3.0.BUILD-SNAPSHOT")
// ...
}
通常你可以遵循Section 63.7, “Discover built-in options for external properties”关于@ConfigurationProperties
(此处主要是ServerProperties和ServerProperties.Undertow),但也要看下EmbeddedServletContainerCustomizer
。一旦获取到UndertowEmbeddedServletContainerFactory
,你就可以使用一个UndertowBuilderCustomizer
修改Undertow的配置以满足你的需求。或更彻底地就是添加你自己的UndertowEmbeddedServletContainerFactory
。
往UndertowEmbeddedServletContainerFactory
添加一个UndertowBuilderCustomizer
,然后添加一个监听者到Builder
:
@Bean
public UndertowEmbeddedServletContainerFactory embeddedServletContainerFactory() {
UndertowEmbeddedServletContainerFactory factory = new UndertowEmbeddedServletContainerFactory();
factory.addBuilderCustomizers(new UndertowBuilderCustomizer() {
@Override
public void customize(Builder builder) {
builder.addHttpListener(8080, "0.0.0.0");
}
});
return factory;
}
Tomcat7可用于Spring Boot,但默认使用的是Tomcat8。如果不能使用Tomcat8(例如,你使用的是Java1.6),你需要改变classpath去引用Tomcat7。
如果正在使用starter pom和parent,你只需要改变Tomcat的version属性,比如,对于一个简单的webapp或service:
<properties>
<tomcat.version>7.0.59tomcat.version>
properties>
<dependencies>
...
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-webartifactId>
dependency>
...
dependencies>
你可以通过设置tomcat.version
属性改变Tomcat的版本:
ext['tomcat.version'] = '7.0.59'
dependencies {
compile 'org.springframework.boot:spring-boot-starter-web'
}
Jetty8可用于Spring Boot,但默认使用的是Jetty9。如果不能使用Jetty9(例如,因为你使用的是Java1.6),你只需改变classpath去引用Jetty8。你也需要排除Jetty的WebSocket相关的依赖。
如果正在使用starter pom和parent,你只需添加Jetty starter,去掉WebSocket依赖,并改变version属性,比如,对于一个简单的webapp或service:
<properties>
<jetty.version>8.1.15.v20140411jetty.version>
<jetty-jsp.version>2.2.0.v201112011158jetty-jsp.version>
properties>
<dependencies>
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-webartifactId>
<exclusions>
<exclusion>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-tomcatartifactId>
exclusion>
exclusions>
dependency>
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-jettyartifactId>
<exclusions>
<exclusion>
<groupId>org.eclipse.jetty.websocketgroupId>
<artifactId>*artifactId>
exclusion>
exclusions>
dependency>
dependencies>
你可以设置jetty.version
属性并排除相关的WebSocket依赖,比如对于一个简单的webapp或service:
ext['jetty.version'] = '8.1.15.v20140411'
dependencies {
compile ('org.springframework.boot:spring-boot-starter-web') {
exclude group: 'org.springframework.boot', module: 'spring-boot-starter-tomcat'
}
compile ('org.springframework.boot:spring-boot-starter-jetty') {
exclude group: 'org.eclipse.jetty.websocket'
}
}
如果想在一个使用内嵌容器的Spring Boot应用中使用@ServerEndpoint,你需要声明一个单独的ServerEndpointExporter @Bean:
@Bean
public ServerEndpointExporter serverEndpointExporter() {
return new ServerEndpointExporter();
}
该bean将用底层的WebSocket容器注册任何的被@ServerEndpoint
注解的beans。当部署到一个单独的servlet容器时,该角色将被一个servlet容器初始化方法履行,ServerEndpointExporter bean也就不是必需的了。
Spring Boot提供两种启用HTTP压缩的机制;一种是Tomcat特有的,另一种是使用一个filter,可以配合Jetty,Tomcat和Undertow。
Tomcat对HTTP响应压缩提供内建支持。默认是禁用的,但可以通过application.properties轻松的启用:
server.tomcat.compression: on
当设置为on
时,Tomcat将压缩响应的长度至少为2048字节。你可以配置一个整型值来设置该限制而不只是on
,比如:
server.tomcat.compression: 4096
默认情况下,Tomcat只压缩某些MIME类型的响应(text/html,text/xml和text/plain)。你可以使用server.tomcat.compressableMimeTypes
属性进行自定义,比如:
server.tomcat.compressableMimeTypes=application/json,application/xml
如果你正在使用Jetty或Undertow,或想要更精确的控制HTTP响应压缩,Spring Boot为Jetty的GzipFilter提供自动配置。虽然该过滤器是Jetty的一部分,但它也兼容Tomcat和Undertow。想要启用该过滤器,只需简单的为你的应用添加org.eclipse.jetty:jetty-servlets
依赖。
GzipFilter可以使用spring.http.gzip.*
属性进行配置。具体参考GzipFilterProperties。
在Spring Boot应用中,任何Spring @RestController
默认应该渲染为JSON响应,只要classpath下存在Jackson2。例如:
@RestController
public class MyController {
@RequestMapping("/thing")
public MyThing thing() {
return new MyThing();
}
}
只要MyThing能够通过Jackson2序列化(比如,一个标准的POJO或Groovy对象),localhost:8080/thing默认响应一个JSON表示。有时在一个浏览器中你可能看到XML响应因为浏览器倾向于发送XML
响应头。
如果classpath下存在Jackson XML扩展(jackson-dataformat-xml),它会被用来渲染XML响应,示例和JSON的非常相似。想要使用它,只需为你的项目添加以下的依赖:
<dependency>
<groupId>com.fasterxml.jackson.dataformatgroupId>
<artifactId>jackson-dataformat-xmlartifactId>
dependency>
你可能也想添加对Woodstox的依赖。它比JDK提供的默认Stax实现快很多,并且支持良好的格式化输出,提高了namespace处理能力:
<dependency>
<groupId>org.codehaus.woodstoxgroupId>
<artifactId>woodstox-core-aslartifactId>
dependency>
如果Jackson的XML扩展不可用,Spring Boot将使用JAXB(JDK默认提供),不过你需要为MyThing添加额外的注解@XmlRootElement
:
@XmlRootElement
public class MyThing {
private String name;
// .. getters and setters
}
想要服务器渲染XML而不是JSON,你可能需要发送一个Accept: text/xml
头部(或使用浏览器)。
在一个HTTP交互中,Spring MVC(客户端和服务端)使用HttpMessageConverters协商内容转换。如果classpath下存在Jackson,你就已经获取到Jackson2ObjectMapperBuilder提供的默认转换器。
创建的ObjectMapper(或用于Jackson XML转换的XmlMapper)实例默认有以下自定义属性:
MapperFeature.DEFAULT_VIEW_INCLUSION
禁用DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES
禁用Spring Boot也有一些简化自定义该行为的特性。
你可以使用当前的environment配置ObjectMapper和XmlMapper实例。Jackson提供一个扩展套件,可以用来简单的关闭或开启一些特性,你可以用它们配置Jackson处理的不同方面。这些特性在Jackson中使用5个枚举进行描述的,并被映射到environment的属性上:
Jackson枚举 | Environment属性 |
---|---|
com.fasterxml.jackson.databind.DeserializationFeature |
`spring.jackson.deserialization.=true |
com.fasterxml.jackson.core.JsonGenerator.Feature |
`spring.jackson.generator.=true |
com.fasterxml.jackson.databind.MapperFeature |
`spring.jackson.mapper.=true |
com.fasterxml.jackson.core.JsonParser.Feature |
`spring.jackson.parser.=true |
com.fasterxml.jackson.databind.SerializationFeature |
`spring.jackson.serialization.=true |
例如,设置spring.jackson.serialization.indent_output=true
可以开启漂亮打印。注意,由于松绑定的使用,indent_output
不必匹配对应的枚举常量INDENT_OUTPUT
。
如果想彻底替换默认的ObjectMapper,你需要定义一个该类型的@Bean
并将它标记为@Primary
。
定义一个Jackson2ObjectMapperBuilder类型的@Bean
将允许你自定义默认的ObjectMapper和XmlMapper(分别用于MappingJackson2HttpMessageConverter和MappingJackson2XmlHttpMessageConverter)。
另一种自定义Jackson的方法是向你的上下文添加com.fasterxml.jackson.databind.Module
类型的beans。它们会被注册入每个ObjectMapper类型的bean,当为你的应用添加新特性时,这就提供了一种全局机制来贡献自定义模块。
最后,如果你提供任何MappingJackson2HttpMessageConverter类型的@Beans
,那它们将替换MVC配置中的默认值。同时,也提供一个HttpMessageConverters类型的bean,它有一些有用的方法可以获取默认的和用户增强的message转换器。
想要获取更多细节可查看Section 65.4, “Customize the @ResponseBody rendering”和WebMvcAutoConfiguration源码。
Spring使用HttpMessageConverters渲染@ResponseBody
(或来自@RestController
的响应)。你可以通过在Spring Boot上下文中添加该类型的beans来贡献其他的转换器。如果你添加的bean类型默认已经包含了(像用于JSON转换的MappingJackson2HttpMessageConverter),那它将替换默认的。Spring Boot提供一个方便的HttpMessageConverters类型的bean,它有一些有用的方法可以访问默认的和用户增强的message转换器(有用,比如你想要手动将它们注入到一个自定义的RestTemplate
)。
在通常的MVC用例中,任何你提供的WebMvcConfigurerAdapter beans通过覆盖configureMessageConverters方法也能贡献转换器,但不同于通常的MVC,你可以只提供你需要的转换器(因为Spring Boot使用相同的机制来贡献它默认的转换器)。最终,如果你通过提供自己的@EnableWebMvc
注解覆盖Spring Boot默认的MVC配置,那你就可以完全控制,并使用来自WebMvcConfigurationSupport的getMessageConverters手动做任何事。
具体参考WebMvcAutoConfiguration源码。
Spring Boot采用Servlet 3 javax.servlet.http.Part
API来支持文件上传。默认情况下,Spring Boot配置Spring MVC在单个请求中每个文件最大1Mb,最多10Mb的文件数据。你可以覆盖那些值,也可以设置临时文件存储的位置(比如,存储到/tmp
文件夹下)及传递数据刷新到磁盘的阀值(通过使用MultipartProperties类暴露的属性)。如果你需要设置文件不受限制,例如,可以设置multipart.maxFileSize
属性值为-1
。
当你想要接收部分(multipart)编码文件数据作为Spring MVC控制器(controller)处理方法中被@RequestParam
注解的MultipartFile类型的参数时,multipart支持就非常有用了。
具体参考MultipartAutoConfiguration源码。
Spring Boot想要服务来自应用程序root /
下的所有内容。如果你想将自己的servlet映射到该目录下也是可以的,但当然你可能失去一些Boot MVC特性。为了添加你自己的servlet,并将它映射到root资源,你只需声明一个Servlet类型的@Bean
,并给它特定的bean名称dispatcherServlet
(如果只想关闭但不替换它,你可以使用该名称创建不同类型的bean)。
完全控制MVC配置的最简单方式是提供你自己的被@EnableWebMvc
注解的@Configuration
。这样所有的MVC配置都逃不出你的掌心。
ViewResolver是Spring MVC的核心组件,它负责转换@Controller
中的视图名称到实际的View实现。注意ViewResolvers主要用在UI应用中,而不是REST风格的服务(View不是用来渲染@ResponseBody
的)。Spring有很多你可以选择的ViewResolver实现,并且Spring自己对如何选择相应实现也没发表意见。另一方面,Spring Boot会根据classpath上的依赖和应用上下文为你安装一或两个ViewResolver实现。DispatcherServlet使用所有在应用上下文中找到的解析器(resolvers),并依次尝试每一个直到它获取到结果,所以如果你正在添加自己的解析器,那就要小心顺序和你的解析器添加的位置。
WebMvcAutoConfiguration将会为你的上下文添加以下ViewResolvers:
defaultViewResolver
的InternalResourceViewResolver。这个会定位可以使用DefaultServlet渲染的物理资源(比如,静态资源和JSP页面)。它在视图(view name)上应用了一个前缀和后缀(默认都为空,但你可以通过spring.view.prefix
和spring.view.suffix
外部配置设置),然后查找在servlet上下文中具有该路径的物理资源。可以通过提供相同类型的bean覆盖它。beanNameViewResolver
的BeanNameViewResolver。这是视图解析器链的一个非常有用的成员,它可以在View被解析时收集任何具有相同名称的beans。viewResolver
的ContentNegotiatingViewResolver只会在实际View类型的beans出现时添加。这是一个’主’解析器,它的职责会代理给其他解析器,它会尝试找到客户端发送的一个匹配’Accept’的HTTP头部。这有一篇有用的,关于你需要更多了解的ContentNegotiatingViewResolver的博客,也要具体查看下源码。通过定义一个名叫’viewResolver’的bean,你可以关闭自动配置的ContentNegotiatingViewResolver。thymeleafViewResolver
的ThymeleafViewResolver。它会通过加前缀和后缀的视图名来查找资源(外部配置为spring.thymeleaf.prefix
和spring.thymeleaf.suffix
,对应的默认为’classpath:/templates/’和’.html’)。你可以通过提供相同名称的bean来覆盖它。freeMarkerViewResolver
的FreeMarkerViewResolver。它会使用加前缀和后缀(外部配置为spring.freemarker.prefix
和spring.freemarker.suffix
,对应的默认值为空和’.ftl’)的视图名从加载路径(外部配置为spring.freemarker.templateLoaderPath
,默认为’classpath:/templates/’)下查找资源。你可以通过提供一个相同名称的bean来覆盖它。groovyTemplateViewResolver
的Groovy TemplateViewResolver。它会使用加前缀和后缀(外部属性为spring.groovy.template.prefix
和spring.groovy.template.suffix
,对应的默认值为’classpath:/templates/’和’.tpl’)的视图名从加载路径下查找资源。你可以通过提供一个相同名称的bean来覆盖它。velocityViewResolver
的VelocityViewResolver。它会使用加前缀和后缀(外部属性为spring.velocity.prefix
和spring.velocity.suffix
,对应的默认值为空和’.vm’)的视图名从加载路径(外部属性为spring.velocity.resourceLoaderPath
,默认为’classpath:/templates/’)下查找资源。你可以通过提供一个相同名称的bean来覆盖它。具体参考: WebMvcAutoConfiguration,ThymeleafAutoConfiguration,FreeMarkerAutoConfiguration,GroovyTemplateAutoConfiguration,VelocityAutoConfiguration。
Spring Boot除了commons-logging API外没有其他强制性的日志依赖,你有很多可选的日志实现。想要使用Logback,你需要包含它,及一些对classpath下commons-logging的绑定。最简单的方式是通过依赖spring-boot-starter-logging
的starter pom。对于一个web应用程序,你只需添加spring-boot-starter-web
依赖,因为它依赖于logging starter。例如,使用Maven:
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-webartifactId>
dependency>
Spring Boot有一个LoggingSystem抽象,用于尝试通过classpath上下文配置日志系统。如果Logback可用,则首选它。如果你唯一需要做的就是设置不同日志的级别,那可以通过在application.properties中使用logging.level
前缀实现,比如:
logging.level.org.springframework.web: DEBUG
logging.level.org.hibernate: ERROR
你也可以使用logging.file
设置日志文件的位置(除控制台之外,默认会输出到控制台)。
想要对日志系统进行更细粒度的配置,你需要使用正在说的LoggingSystem支持的原生配置格式。默认情况下,Spring Boot从系统的默认位置加载原生配置(比如对于Logback为classpath:logback.xml
),但你可以使用logging.config
属性设置配置文件的位置。
如果你将一个logback.xml放到classpath根目录下,那它将会被从这加载。Spring Boot提供一个默认的基本配置,如果你只是设置日志级别,那你可以包含它,比如:
<configuration>
<include resource="org/springframework/boot/logging/logback/base.xml"/>
<logger name="org.springframework.web" level="DEBUG"/>
configuration>
如果查看spring-boot jar包中的默认logback.xml,你将会看到LoggingSystem为你创建的很多有用的系统属性,比如:
- PID,当前进程id− {LOG_FILE},如果在Boot外部配置中设置了logging.file
- ${LOG_PATH},如果设置了logging.path
(表示日志文件产生的目录)
Spring Boot也提供使用自定义的Logback转换器在控制台上输出一些漂亮的彩色ANSI日志信息(不是日志文件)。具体参考默认的base.xml
配置。
如果Groovy在classpath下,你也可以使用logback.groovy配置Logback。
Spring Boot也支持Log4j或Log4j 2作为日志配置,但只有在它们中的某个在classpath下存在的情况。如果你正在使用starter poms进行依赖装配,这意味着你需要排除Logback,然后包含你选择的Log4j版本。如果你不使用starter poms,那除了你选择的Log4j版本外还要提供commons-logging(至少)。
最简单的方式可能就是通过starter poms,尽管它需要排除一些依赖,比如,在Maven中:
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-webartifactId>
dependency>
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starterartifactId>
<exclusions>
<exclusion>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-loggingartifactId>
exclusion>
exclusions>
dependency>
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-log4jartifactId>
dependency>
想要使用Log4j 2,只需要依赖spring-boot-starter-log4j2
而不是spring-boot-starter-log4j
。
注:使用Log4j各版本的starters都会收集好依赖以满足common logging的要求(比如,Tomcat中使用java.util.logging
,但使用Log4j或 Log4j 2作为输出)。具体查看Actuator Log4j或Log4j 2的示例,了解如何将它用于实战。
除了它的默认XML配置格式,Log4j 2也支持YAML和JSON配置文件。想要使用其他配置文件格式来配置Log4j 2,你需要添加合适的依赖到classpath。为了使用YAML,你需要添加com.fasterxml.jackson.dataformat:jackson-dataformat-yaml
依赖,Log4j 2将查找名称为log4j2.yaml
或log4j2.yml
的配置文件。为了使用JSON,你需要添加com.fasterxml.jackson.core:jackson-databind
依赖,Log4j 2将查找名称为log4j2.json
或log4j2.jsn
的配置文件
想要覆盖默认的设置只需要定义一个你自己的DataSource类型的@Bean
。Spring Boot提供一个工具构建类DataSourceBuilder,可用来创建一个标准的DataSource(如果它处于classpath下),或者仅创建你自己的DataSource,然后将它和在Section 23.7.1, “Third-party configuration”解释的一系列Environment属性绑定。
比如:
@Bean
@ConfigurationProperties(prefix="datasource.mine")
public DataSource dataSource() {
return new FancyDataSource();
}
datasource.mine.jdbcUrl=jdbc:h2:mem:mydb
datasource.mine.user=sa
datasource.mine.poolSize=30
具体参考’Spring Boot特性’章节中的Section 28.1, “Configure a DataSource”和DataSourceAutoConfiguration类源码。
创建多个数据源和创建第一个工作都是一样的。如果使用针对JDBC或JPA的默认自动配置,你可能想要将其中一个设置为@Primary
(然后它就能被任何@Autowired
注入获取)。
@Bean
@Primary
@ConfigurationProperties(prefix="datasource.primary")
public DataSource primaryDataSource() {
return DataSourceBuilder.create().build();
}
@Bean
@ConfigurationProperties(prefix="datasource.secondary")
public DataSource secondaryDataSource() {
return DataSourceBuilder.create().build();
}
Spring Data可以为你的@Repository
接口创建各种风格的实现。Spring Boot会为你处理所有事情,只要那些@Repositories
接口跟你的@EnableAutoConfiguration
类处于相同的包(或子包)。
对于很多应用来说,你需要做的就是将正确的Spring Data依赖添加到classpath下(对于JPA有一个spring-boot-starter-data-jpa
,对于Mongodb有一个spring-boot-starter-data-mongodb
),创建一些repository接口来处理@Entity
对象。具体参考JPA sample或Mongodb sample。
Spring Boot会基于它找到的@EnableAutoConfiguration
来尝试猜测你的@Repository
定义的位置。想要获取更多控制,可以使用@EnableJpaRepositories
注解(来自Spring Data JPA)。
@Entity
定义Spring Boot会基于它找到的@EnableAutoConfiguration
来尝试猜测你的@Entity
定义的位置。想要获取更多控制,你可以使用@EntityScan
注解,比如:
@Configuration
@EnableAutoConfiguration
@EntityScan(basePackageClasses=City.class)
public class Application {
//...
}
Spring Data JPA已经提供了一些独立的配置选项(比如,针对SQL日志),并且Spring Boot会暴露它们,针对hibernate的外部配置属性也更多些。最常见的选项如下:
spring.jpa.hibernate.ddl-auto: create-drop
spring.jpa.hibernate.naming_strategy: org.hibernate.cfg.ImprovedNamingStrategy
spring.jpa.database: H2
spring.jpa.show-sql: true
(由于宽松的数据绑定策略,连字符或下划线作为属性keys作用应该是等效的)ddl-auto
配置是个特殊情况,它有不同的默认设置,这取决于你是否使用一个内嵌数据库(create-drop)。当本地EntityManagerFactory被创建时,所有spring.jpa.properties.*
属性都被作为正常的JPA属性(去掉前缀)传递进去了。
具体参考HibernateJpaAutoConfiguration和JpaBaseConfiguration。
为了完全控制EntityManagerFactory的配置,你需要添加一个名为entityManagerFactory
的@Bean
。Spring Boot自动配置会根据是否存在该类型的bean来关闭它的实体管理器(entity manager)。
即使默认的EntityManagerFactory工作的很好,你也需要定义一个新的EntityManagerFactory,因为一旦出现第二个该类型的bean,默认的将会被关闭。为了轻松的实现该操作,你可以使用Spring Boot提供的EntityManagerBuilder,或者如果你喜欢的话可以直接使用来自Spring ORM的LocalContainerEntityManagerFactoryBean。
示例:
// add two data sources configured as above
@Bean
public LocalContainerEntityManagerFactoryBean customerEntityManagerFactory(
EntityManagerFactoryBuilder builder) {
return builder
.dataSource(customerDataSource())
.packages(Customer.class)
.persistenceUnit("customers")
.build();
}
@Bean
public LocalContainerEntityManagerFactoryBean orderEntityManagerFactory(
EntityManagerFactoryBuilder builder) {
return builder
.dataSource(orderDataSource())
.packages(Order.class)
.persistenceUnit("orders")
.build();
}
上面的配置靠自己基本可以运行。想要完成作品你也需要为两个EntityManagers配置TransactionManagers。其中的一个会被Spring Boot默认的JpaTransactionManager获取,如果你将它标记为@Primary
。另一个需要显式注入到一个新实例。或你可以使用一个JTA事物管理器生成它两个。
Spring不要求使用XML配置JPA提供者(provider),并且Spring Boot假定你想要充分利用该特性。如果你倾向于使用persistence.xml
,那你需要定义你自己的id为’entityManagerFactory’的LocalEntityManagerFactoryBean类型的@Bean
,并在那设置持久化单元的名称。
默认设置可查看JpaBaseConfiguration
Spring Data JPA和Spring Data Mongo都能自动为你创建Repository实现。如果它们同时出现在classpath下,你可能需要添加额外的配置来告诉Spring Boot你想要哪个(或两个)为你创建仓库。最明确地方式是使用标准的Spring Data @Enable*Repositories
,然后告诉它你的Repository接口的位置(此处*即可以是Jpa,也可以是Mongo,或者两者都是)。
这里也有spring.data.*.repositories.enabled
标志,可用来在外部配置中开启或关闭仓库的自动配置。这在你想关闭Mongo仓库,但仍旧使用自动配置的MongoTemplate时非常有用。
相同的障碍和特性也存在于其他自动配置的Spring Data仓库类型(Elasticsearch, Solr)。只需要改变对应注解的名称和标志。
Spring Data REST能够将Repository的实现暴露为REST端点,只要该应用启用Spring MVC。
Spring Boot暴露一系列来自spring.data.rest
命名空间的有用属性来定制化RepositoryRestConfiguration。如果需要提供其他定制,你可以创建一个继承自SpringBootRepositoryRestMvcConfiguration的@Configuration
类。该类功能和RepositoryRestMvcConfiguration相同,但允许你继续使用spring.data.rest.*
属性。
一个数据库可以使用不同的方式进行初始化,这取决于你的技术栈。或者你可以手动完成该任务,只要数据库是单独的过程。
JPA有个生成DDL的特性,这些可以设置为在数据库启动时运行。这可以通过两个外部属性进行控制:
spring.jpa.generate-ddl
(boolean)控制该特性的关闭和开启,跟实现者没关系spring.jpa.hibernate.ddl-auto
(enum)是一个Hibernate特性,用于更细力度的控制该行为。更多详情参考以下内容。
使用Hibernate初始化数据库
你可以显式设置spring.jpa.hibernate.ddl-auto
,标准的Hibernate属性值有none
,validate
,update
,create
,create-drop
。Spring Boot根据你的数据库是否为内嵌数据库来选择相应的默认值,如果是内嵌型的则默认值为create-drop
,否则为none
。通过查看Connection类型可以检查是否为内嵌型数据库,hsqldb,h2和derby是内嵌的,其他都不是。当从内存数据库迁移到一个真正的数据库时,你需要当心,在新的平台中不能对数据库表和数据是否存在进行臆断。你也需要显式设置ddl-auto
,或使用其他机制初始化数据库。
此外,启动时处于classpath根目录下的import.sql文件会被执行。这在demos或测试时很有用,但在生产环境中你可能不期望这样。这是Hibernate的特性,和Spring没有一点关系。
Spring JDBC有一个DataSource初始化特性。Spring Boot默认启用了该特性,并从标准的位置schema.sql和data.sql(位于classpath根目录)加载SQL。此外,Spring Boot将加载schema-${platform}.sql
和data-${platform}.sql
文件(如果存在),在这里platform是spring.datasource.platform
的值,比如,你可以将它设置为数据库的供应商名称(hsqldb, h2, oracle, mysql, postgresql等)。Spring Boot默认启用Spring JDBC初始化快速失败特性,所以如果脚本导致异常产生,那应用程序将启动失败。脚本的位置可以通过设置spring.datasource.schema
和spring.datasource.data
来改变,如果设置spring.datasource.initialize=false
则哪个位置都不会被处理。
你可以设置spring.datasource.continueOnError=true
禁用快速失败特性。一旦应用程序成熟并被部署了很多次,那该设置就很有用,因为脚本可以充当”可怜人的迁移”-例如,插入失败时意味着数据已经存在,也就没必要阻止应用继续运行。
如果你想要在一个JPA应用中使用schema.sql,那如果Hibernate试图创建相同的表,ddl-auto=create-drop
将导致错误产生。为了避免那些错误,可以将ddl-auto
设置为“”(推荐)或“none”。不管是否使用ddl-auto=create-drop
,你总可以使用data.sql初始化新数据。
如果你正在使用Spring Batch,那么它会为大多数的流行数据库平台预装SQL初始化脚本。Spring Boot会检测你的数据库类型,并默认执行那些脚本,在这种情况下将关闭快速失败特性(错误被记录但不会阻止应用启动)。这是因为那些脚本是可信任的,通常不会包含bugs,所以错误会被忽略掉,并且对错误的忽略可以让脚本具有幂等性。你可以使用spring.batch.initializer.enabled=false
显式关闭初始化功能。
Spring Boot跟高级别的数据迁移工具Flyway(基于SQL)和Liquibase(XML)工作的很好。通常我们倾向于Flyway,因为它一眼看去好像很容易,另外它通常不需要平台独立:一般一个或至多需要两个平台。
想要在启动时自动运行Flyway数据库迁移,需要将org.flywaydb:flyway-core
添加到你的classpath下。
迁移是一些V
格式的脚本(
是一个下划线分割的版本号,比如’1’或’2_1’)。默认情况下,它们存放在一个classpath:db/migration
的文件夹中,但你可以使用flyway.locations
(一个列表)来改变它。详情可参考flyway-core中的Flyway类,查看一些可用的配置,比如schemas。Spring Boot在FlywayProperties中提供了一个小的属性集,可用于禁止迁移,或关闭位置检测。
默认情况下,Flyway将自动注入(@Primary
)DataSource到你的上下文,并用它进行数据迁移。如果你想使用一个不同的DataSource,你可以创建一个,并将它标记为@FlywayDataSource
的@Bean
-如果你这样做了,且想要两个数据源,记得创建另一个并将它标记为@Primary
。或者你可以通过在外部配置文件中设置flyway.[url,user,password]
来使用Flyway的原生DataSource。
这是一个Flyway示例,你可以作为参考。
想要在启动时自动运行Liquibase数据库迁移,你需要将org.liquibase:liquibase-core
添加到classpath下。
主改变日志(master change log)默认从db/changelog/db.changelog-master.yaml
读取,但你可以使用liquibase.change-log
进行设置。详情查看LiquibaseProperties以获取可用设置,比如上下文,默认的schema等。
这里有个Liquibase示例可作为参考。
你可以在上下文的某个地方添加@EnableBatchProcessing
来启用Spring Batch的自动配置功能。
默认情况下,在启动时它会执行应用的所有作业(Jobs),具体查看JobLauncherCommandLineRunner。你可以通过指定spring.batch.job.names
(多个作业名以逗号分割)来缩小到一个特定的作业或多个作业。
如果应用上下文包含一个JobRegistry,那么处于spring.batch.job.names
中的作业将会从registry中查找,而不是从上下文中自动装配。这是复杂系统中常见的一个模式,在这些系统中多个作业被定义在子上下文和注册中心。
具体参考BatchAutoConfiguration和@EnableBatchProcessing。
在一个单独的应用中,执行器的HTTP端口默认和主HTTP端口相同。想要让应用监听不同的端口,你可以设置外部属性management.port
。为了监听一个完全不同的网络地址(比如,你有一个用于管理的内部网络和一个用于用户应用程序的外部网络),你可以将management.address
设置为一个可用的IP地址,然后将服务器绑定到该地址。
查看ManagementServerProperties源码和’Production-ready特性’章节中的Section 41.3, “Customizing the management server port”来获取更多详情。
Spring Boot安装了一个’whitelabel’错误页面,如果你遇到一个服务器错误(机器客户端消费的是JSON,其他媒体类型则会看到一个具有正确错误码的合乎情理的响应),那就能在客户端浏览器中看到该页面。你可以设置error.whitelabel.enabled=false
来关闭该功能,但通常你想要添加自己的错误页面来取代whitelabel。确切地说,如何实现取决于你使用的模板技术。例如,你正在使用Thymeleaf,你将添加一个error.html模板。如果你正在使用FreeMarker,那你将添加一个error.ftl模板。通常,你需要的只是一个名称为error的View,和/或一个处理/error
路径的@Controller
。除非你替换了一些默认配置,否则你将在你的ApplicationContext中找到一个BeanNameViewResolver,所以一个id为error的@Bean
可能是完成该操作的一个简单方式。详情参考ErrorMvcAutoConfiguration。
查看Error Handling章节,了解下如何将处理器(handlers)注册到servlet容器中。
不管你在应用的什么地方定义了一个使用@EnableWebSecurity
注解的@Configuration
,它将会关闭Spring Boot中的默认webapp安全设置。想要调整默认值,你可以尝试设置security.*
属性(具体查看SecurityProperties和常见应用属性的SECURITY章节)。
如果你提供了一个AuthenticationManager类型的@Bean
,那么默认的就不会被创建了,所以你可以获得Spring Security可用的全部特性(比如,不同的认证选项)。
Spring Security也提供了一个方便的AuthenticationManagerBuilder,可用于构建具有常见选项的AuthenticationManager。在一个webapp中,推荐将它注入到WebSecurityConfigurerAdapter的一个void方法中,比如:
@Configuration
public class SecurityConfiguration extends WebSecurityConfigurerAdapter {
@Autowired
public void configureGlobal(AuthenticationManagerBuilder auth) throws Exception {
auth.inMemoryAuthentication()
.withUser("barry").password("password").roles("USER"); // ... etc.
}
// ... other stuff for application security
}
如果把它放到一个内部类或一个单独的类中,你将得到最好的结果(也就是不跟很多其他@Beans
混合在一起将允许你改变实例化的顺序)。secure web sample是一个有用的参考模板。
如果你遇到了实例化问题(比如,使用JDBC或JPA进行用户详细信息的存储),那将AuthenticationManagerBuilder回调提取到一个GlobalAuthenticationConfigurerAdapter(放到init()方法内以防其他地方也需要authentication manager)可能是个不错的选择,比如:
@Configuration
public class AuthenticationManagerConfiguration extends
GlobalAuthenticationConfigurerAdapter {
@Override
public void init(AuthenticationManagerBuilder auth) {
auth.inMemoryAuthentication() // ... etc.
}
}
对于任何应用来说,确保所有的主端点(URL)都只在HTTPS下可用是个重要的苦差事。如果你使用Tomcat作为servlet容器,那Spring Boot如果发现一些环境设置的话,它将自动添加Tomcat自己的RemoteIpValve,你也可以依赖于HttpServletRequest来报告是否请求是安全的(即使代理服务器的downstream处理真实的SSL终端)。这个标准行为取决于某些请求头是否出现(x-forwarded-for
和x-forwarded-proto
),这些请求头的名称都是约定好的,所以对于大多数前端和代理都是有效的。
你可以向application.properties添加以下设置里开启该功能,比如:
server.tomcat.remote_ip_header=x-forwarded-for
server.tomcat.protocol_header=x-forwarded-proto
(这些属性出现一个就会开启该功能,或者你可以通过添加一个TomcatEmbeddedServletContainerFactory bean自己添加RemoteIpValve)
Spring Security也可以配置成针对所以或某些请求需要一个安全渠道(channel)。想要在一个Spring Boot应用中开启它,你只需将application.properties中的security.require_ssl
设置为true
即可。
Spring Boot有很多用于热加载的选项。使用IDE开发是一个不错的方式,特别是需要调试的时候(所有的现代IDEs都允许重新加载静态资源,通常也支持对变更的Java类进行热交换)。Maven和Gradle插件也支持命令行下的静态文件热加载。如果你使用其他高级工具编写css/js,并使用外部的css/js编译器,那你就可以充分利用该功能。
如果你正在使用Thymeleaf,那就将spring.thymeleaf.cache
设置为false。查看ThymeleafAutoConfiguration可以获取其他Thymeleaf自定义选项。
如果你正在使用FreeMarker,那就将spring.freemarker.cache
设置为false。查看FreeMarkerAutoConfiguration 可以获取其他FreeMarker自定义选项。
如果你正在使用Groovy模板,那就将spring.groovy.template.cache
设置为false。查看GroovyTemplateAutoConfiguration可以获取其他Groovy自定义选项。
如果你正在使用Velocity,那就将spring.velocity.cache
设置为false。查看VelocityAutoConfiguration可以获取其他Velocity自定义选项。
现代IDEs(Eclipse, IDEA等)都支持字节码的热交换,所以如果你做了一个没有影响类或方法签名的改变,它会利索地重新加载并没有任何影响。
Spring Loaded在这方面走的更远,它能够重新加载方法签名改变的类定义。如果对它进行一些自定义配置可以强制ApplicationContext刷新自己(但没有通用的机制来确保这对一个运行中的应用总是安全的,所以它可能只是一个开发时间的技巧)。
为了在Maven命令行下使用Spring Loaded,你只需将它作为一个依赖添加到Spring Boot插件声明中即可,比如:
<plugin>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-maven-pluginartifactId>
<dependencies>
<dependency>
<groupId>org.springframeworkgroupId>
<artifactId>springloadedartifactId>
<version>1.2.0.RELEASEversion>
dependency>
dependencies>
plugin>
正常情况下,这在Eclipse和IntelliJ中工作的相当漂亮,只要它们有相应的,和Maven默认一致的构建配置(Eclipse m2e对此支持的更好,开箱即用)。
如果想将Spring Loaded和Gradle,IntelliJ结合起来,那你需要付出代价。默认情况下,IntelliJ将类编译到一个跟Gradle不同的位置,这会导致Spring Loaded监控失败。
为了正确配置IntelliJ,你可以使用idea
Gradle插件:
buildscript {
repositories { jcenter() }
dependencies {
classpath "org.springframework.boot:spring-boot-gradle-plugin:1.3.0.BUILD-SNAPSHOT"
classpath 'org.springframework:springloaded:1.2.0.RELEASE'
}
}
apply plugin: 'idea'
idea {
module {
inheritOutputDirs = false
outputDir = file("$buildDir/classes/main/")
}
}
// ...
注:IntelliJ必须配置跟命令行Gradle任务相同的Java版本,并且springloaded必须作为一个buildscript依赖被包含进去。
此外,你也可以启用Intellij内部的Make Project Automatically
,这样不管什么时候只要文件被保存都会自动编译你的代码。
如果你使用Maven进行一个直接或间接继承spring-boot-dependencies
(比如spring-boot-starter-parent
)的构建,并想覆盖一个特定的第三方依赖,那你可以添加合适的
元素。浏览spring-boot-dependencies POM可以获取一个全面的属性列表。例如,想要选择一个不同的slf4j版本,你可以添加以下内容:
<properties>
<slf4j.version>1.7.5<slf4j.version>
properties>
注:这只在你的Maven项目继承(直接或间接)自spring-boot-dependencies
才有用。如果你使用
,将spring-boot-dependencies
添加到自己的dependencyManagement
片段,那你必须自己重新定义artifact而不是覆盖属性。
注:每个Spring Boot发布都是基于一些特定的第三方依赖集进行设计和测试的,覆盖版本可能导致兼容性问题。
spring-boot-maven-plugin
能够用来创建可执行的’胖’JAR。如果你正在使用spring-boot-starter-parent
POM,你可以简单地声明该插件,然后你的jar将被重新打包:
<build>
<plugins>
<plugin>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-maven-pluginartifactId>
plugin>
plugins>
build>
如果没有使用parent POM,你仍旧可以使用该插件。不过,你需要另外添加一个
片段:
<build>
<plugins>
<plugin>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-maven-pluginartifactId>
<version>1.3.0.BUILD-SNAPSHOTversion>
<executions>
<execution>
<goals>
<goal>repackagegoal>
goals>
execution>
executions>
plugin>
plugins>
build>
查看插件文档获取详细的用例。
如果你想将自己的项目以library jar的形式被其他项目依赖,并且需要它是一个可执行版本(例如demo),你需要使用略微不同的方式来配置该构建。
对于Maven来说,正常的JAR插件和Spring Boot插件都有一个’classifier’,你可以添加它来创建另外的JAR。示例如下(使用Spring Boot Starter Parent管理插件版本,其他配置采用默认设置):
<build>
<plugins>
<plugin>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-maven-pluginartifactId>
<configuration>
<classifier>execclassifier>
configuration>
plugin>
plugins>
build>
上述配置会产生两个jars,默认的一个和使用带有classifier ‘exec’的Boot插件构建的可执行的一个。
对于Gradle用户来说,步骤类似。示例如下:
bootRepackage {
classifier = 'exec'
}
在一个可执行jar中,为了运行,多数内嵌的库不需要拆包(unpacked),然而有一些库可能会遇到问题。例如,JRuby包含它自己的内嵌jar,它假定jruby-complete.jar
本身总是能够直接作为文件访问的。
为了处理任何有问题的库,你可以标记那些特定的内嵌jars,让它们在可执行jar第一次运行时自动解压到一个临时文件夹中。例如,为了将JRuby标记为使用Maven插件拆包,你需要添加如下的配置:
<build>
<plugins>
<plugin>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-maven-pluginartifactId>
<configuration>
<requiresUnpack>
<dependency>
<groupId>org.jrubygroupId>
<artifactId>jruby-completeartifactId>
dependency>
requiresUnpack>
configuration>
plugin>
plugins>
build>
使用Gradle完全上述操作:
springBoot {
requiresUnpack = ['org.jruby:jruby-complete']
}
如果你构建的产物既有可执行的jar和非可执行的jar,那你常常需要为可执行的版本添加额外的配置文件,而这些文件在一个library jar中是不需要的。比如,application.yml配置文件可能需要从非可执行的JAR中排除。
下面是如何在Maven中实现:
<build>
<plugins>
<plugin>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-maven-pluginartifactId>
<configuration>
<classifier>execclassifier>
configuration>
plugin>
<plugin>
<artifactId>maven-jar-pluginartifactId>
<executions>
<execution>
<id>execid>
<phase>packagephase>
<goals>
<goal>jargoal>
goals>
<configuration>
<classifier>execclassifier>
configuration>
execution>
<execution>
<phase>packagephase>
<goals>
<goal>jargoal>
goals>
<configuration>
<forceCreation>trueforceCreation>
<excludes>
<exclude>application.ymlexclude>
excludes>
configuration>
execution>
executions>
plugin>
plugins>
build>
在Gradle中,你可以使用标准任务的DSL(领域特定语言)特性创建一个新的JAR存档,然后在bootRepackage任务中使用withJarTask属性添加对它的依赖:
jar {
baseName = 'spring-boot-sample-profile'
version = '0.0.0'
excludes = ['**/application.yml']
}
task('execJar', type:Jar, dependsOn: 'jar') {
baseName = 'spring-boot-sample-profile'
version = '0.0.0'
classifier = 'exec'
from sourceSets.main.output
}
bootRepackage {
withJarTask = tasks['execJar']
}
想要为使用Maven启动的Spring Boot应用添加一个远程调试器,你可以使用mave插件的jvmArguments属性。详情参考示例。
想要为使用Gradle启动的Spring Boot应用添加一个远程调试器,你可以使用build.gradle的applicationDefaultJvmArgs属性或--debug-jvm
命令行选项。
build.gradle:
applicationDefaultJvmArgs = [
"-agentlib:jdwp=transport=dt_socket,server=y,suspend=y,address=5005"
]
命令行:
$ gradle run --debug-jvm
详情查看Gradle应用插件。
想要使用Ant进行构建,你需要抓取依赖,编译,然后像通常那样创建一个jar或war存档。为了让它可以执行:
Main-Class
,比如对于jar文件使用JarLauncher,然后将其他需要的属性以manifest实体指定,主要是一个Start-Class
。provided
(内嵌容器)依赖添加到一个内嵌的lib-provided
目录。记住不要压缩存档中的实体。spring-boot-loader
类(这样Main-Class
就可用了)。示例:
<target name="build" depends="compile">
<copy todir="target/classes/lib">
<fileset dir="lib/runtime" />
copy>
<jar destfile="target/spring-boot-sample-actuator-${spring-boot.version}.jar" compress="false">
<fileset dir="target/classes" />
<fileset dir="src/main/resources" />
<zipfileset src="lib/loader/spring-boot-loader-jar-${spring-boot.version}.jar" />
<manifest>
<attribute name="Main-Class" value="org.springframework.boot.loader.JarLauncher" />
<attribute name="Start-Class" value="${start-class}" />
manifest>
jar>
target>
该Actuator示例中有一个build.xml文件,可以使用以下命令来运行:
$ ant -lib /ivy-2.2.jar
在上述操作之后,你可以使用以下命令运行该应用:
$ java -jar target/*.jar
如果想在Java6环境中使用Spring Boot,你需要改变一些配置。具体的变化取决于你应用的功能。
如果你在使用Boot的内嵌Servlet容器,你需要使用一个兼容Java6的容器。Tomcat 7和Jetty 8都是Java 6兼容的。具体参考Section 63.15, “Use Tomcat 7”和Section 63.16, “Use Jetty 8”。
Java事务API自身并不要求Java 7,而是官方的API jar包含的已构建类要求Java 7。如果你正在使用JTA,那么你需要使用能够在Java 6工作的构建版本替换官方的JTA 1.2 API jar。为了完成该操作,你需要排除任何对javax.transaction:javax.transaction-api
的传递依赖,并使用org.jboss.spec.javax.transaction:jboss-transaction-api_1.2_spec:1.0.0.Final
依赖替换它们。
产生一个可部署war包的第一步是提供一个SpringBootServletInitializer子类,并覆盖它的configure方法。这充分利用了Spring框架对Servlet 3.0的支持,并允许你在应用通过servlet容器启动时配置它。通常,你只需把应用的主类改为继承SpringBootServletInitializer即可:
@SpringBootApplication
public class Application extends SpringBootServletInitializer {
@Override
protected SpringApplicationBuilder configure(SpringApplicationBuilder application) {
return application.sources(Application.class);
}
public static void main(String[] args) throws Exception {
SpringApplication.run(Application.class, args);
}
}
下一步是更新你的构建配置,这样你的项目将产生一个war包而不是jar包。如果你使用Maven,并使用spring-boot-starter-parent
(为了配置Maven的war插件),所有你需要做的就是更改pom.xml的packaging为war:
<packaging>warpackaging>
如果你使用Gradle,你需要修改build.gradle来将war插件应用到项目上:
apply plugin: 'war'
该过程最后的一步是确保内嵌的servlet容器不能干扰war包将部署的servlet容器。为了达到这个目的,你需要将内嵌容器的依赖标记为provided。
如果使用Maven:
<dependencies>
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-tomcatartifactId>
<scope>providedscope>
dependency>
dependencies>
如果使用Gradle:
dependencies {
// …
providedRuntime 'org.springframework.boot:spring-boot-starter-tomcat'
// …
}
如果你使用Spring Boot构建工具,将内嵌容器依赖标记为provided将产生一个可执行war包,在lib-provided
目录有该war包的provided依赖。这意味着,除了部署到servlet容器,你还可以通过使用命令行java -jar
命令来运行应用。
注:查看Spring Boot基于以上配置的一个Maven示例应用。
老的Servlet容器不支持在Servlet 3.0中使用的ServletContextInitializer启动处理。你仍旧可以在这些容器使用Spring和Spring Boot,但你需要为应用添加一个web.xml,并将它配置为通过一个DispatcherServlet加载一个ApplicationContext。
对于一个非web项目,转换为Spring Boot应用很容易(抛弃创建ApplicationContext的代码,取而代之的是调用SpringApplication或SpringApplicationBuilder)。Spring MVC web应用通常先创建一个可部署的war应用,然后将它迁移为一个可执行的war或jar。建议阅读Getting Started Guide on Converting a jar to a war.。
通过继承SpringBootServletInitializer创建一个可执行war(比如,在一个名为Application的类中),然后添加Spring Boot的@EnableAutoConfiguration
注解。示例:
@Configuration
@EnableAutoConfiguration
@ComponentScan
public class Application extends SpringBootServletInitializer {
@Override
protected SpringApplicationBuilder configure(SpringApplicationBuilder application) {
// Customize the application or call application.sources(...) to add sources
// Since our example is itself a @Configuration class we actually don't
// need to override this method.
return application;
}
}
记住不管你往sources放什么东西,它仅是一个Spring ApplicationContext,正常情况下,任何生效的在这里也会起作用。有一些beans你可以先移除,然后让Spring Boot提供它的默认实现,不过有可能需要先完成一些事情。
静态资源可以移到classpath根目录下的/public
(或/static
,/resources
,/META-INF/resources
)。同样的方式也适合于messages.properties
(Spring Boot在classpath根目录下自动发现这些配置)。
美妙的(Vanilla usage of)Spring DispatcherServlet和Spring Security不需要改变。如果你的应用有其他特性,比如使用其他servlets或filters,那你可能需要添加一些配置到你的Application上下文中,按以下操作替换web.xml的那些元素:
@Bean
,就好像web.xml中的
和
。@Bean
(类似于
和
)。@Import
添加到你的Application中。简单的情况下,大量使用注解配置可以在几行内定义@Bean
定义。一旦war可以使用,我们就通过添加一个main方法到Application来让它可以执行,比如:
public static void main(String[] args) {
SpringApplication.run(Application.class, args);
}
应用可以划分为多个类别:
所有这些都可以进行适当的转化,但每个可能需要稍微不同的技巧。
Servlet 3.0+的应用转化的相当简单,如果它们已经使用Spring Servlet 3.0+初始化器辅助类。通常所有来自一个存在的WebApplicationInitializer的代码可以移到一个SpringBootServletInitializer中。如果一个存在的应用有多个ApplicationContext(比如,如果它使用AbstractDispatcherServletInitializer),那你可以将所有上下文源放进一个单一的SpringApplication。你遇到的主要难题可能是如果那样不能工作,那你就要维护上下文层次。参考示例entry on building a hierarchy。一个存在的包含web相关特性的父上下文通常需要分解,这样所有的ServletContextAware组件都处于子上下文中。
对于还不是Spring应用的应用来说,上面的指南有助于你把应用转换为一个Spring Boot应用,但你也可以选择其他方式。
想要将Spring Boot应用部署到Weblogic,你需要确保你的servlet初始化器直接实现WebApplicationInitializer(即使你继承的基类已经实现了它)。
一个传统的Weblogic初始化器可能如下所示:
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.context.web.SpringBootServletInitializer;
import org.springframework.web.WebApplicationInitializer;
@SpringBootApplication
public class MyApplication extends SpringBootServletInitializer implements WebApplicationInitializer {
}
如果使用logback,你需要告诉Weblogic你倾向使用的打包版本而不是服务器预装的版本。你可以通过添加一个具有如下内容的WEB-INF/weblogic.xml
实现该操作:
<wls:weblogic-web-app
xmlns:wls="http://xmlns.oracle.com/weblogic/weblogic-web-app"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee
http://java.sun.com/xml/ns/javaee/ejb-jar_3_0.xsd
http://xmlns.oracle.com/weblogic/weblogic-web-app
http://xmlns.oracle.com/weblogic/weblogic-web-app/1.4/weblogic-web-app.xsd">
<wls:container-descriptor>
<wls:prefer-application-packages>
<wls:package-name>org.slf4jwls:package-name>
wls:prefer-application-packages>
wls:container-descriptor>
wls:weblogic-web-app>
Spring Boot使用 Servlet 3.0 APIs初始化ServletContext(注册Servlets等),所以你不能在一个Servlet 2.5的容器中原封不动的使用同样的应用。使用一些特定的工具也是可以在一个老的容器中运行Spring Boot应用的。如果添加了org.springframework.boot:spring-boot-legacy
依赖,你只需要创建一个web.xml,声明一个用于创建应用上下文的上下文监听器,过滤器和servlets。上下文监听器是专用于Spring Boot的,其他的都是一个Servlet 2.5的Spring应用所具有的。示例:
<web-app version="2.5" xmlns="http://java.sun.com/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd">
<context-param>
<param-name>contextConfigLocationparam-name>
<param-value>demo.Applicationparam-value>
context-param>
<listener>
<listener-class>org.springframework.boot.legacy.context.web.SpringBootContextLoaderListenerlistener-class>
listener>
<filter>
<filter-name>metricFilterfilter-name>
<filter-class>org.springframework.web.filter.DelegatingFilterProxyfilter-class>
filter>
<filter-mapping>
<filter-name>metricFilterfilter-name>
<url-pattern>/*url-pattern>
filter-mapping>
<servlet>
<servlet-name>appServletservlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServletservlet-class>
<init-param>
<param-name>contextAttributeparam-name>
<param-value>org.springframework.web.context.WebApplicationContext.ROOTparam-value>
init-param>
<load-on-startup>1load-on-startup>
servlet>
<servlet-mapping>
<servlet-name>appServletservlet-name>
<url-pattern>/url-pattern>
servlet-mapping>
web-app>
在该示例中,我们使用一个单一的应用上下文(通过上下文监听器创建的),然后使用一个init参数将它附加到DispatcherServlet。这在一个Spring Boot应用中是很正常的(你通常只有一个应用上下文)。