一、概述

spring-boot-devtools为应用提供一些开发时特性,包括默认值设置,自动重启,livereload等。本文将逐一介绍这些特性,并做演示。



二、如何使用spring-boot-devtools

在pom中,引入spring-boot-devtools即可:

<dependency>
 <groupId>org.springframework.boot</groupId>
 <artifactId>spring-boot-devtools</artifactId>
 <optional>true</optional>
</dependency>

将optional置为true可以避免该引用传递到其他模块。 
Developer tools在运行完整的packaged app时是自动关闭的,即若使用Java –jar时会被当做生产应用。安全起见,可以在maven中增加excludeDevtools编译属性来移除jar包。



三、默认值设置

Spring Boot支持的一些库会使用缓存改善性能。例如,Thymeleaf会缓存末班,以防止重复解析xml源文件。虽然缓存在生产环境很有效,但是在开发时却会影响效率。如果你在IDE中修改了模板文件,缓存会阻止你立即看到修改的结果。 
缓存选项通常由application.properties中的属性配置。例如Thymeleaf提供spring.thymeleaf.cache属性。 
spring-boot-devtools可以帮你自动设置开发时的配置值,免除你手动定义的烦恼。 
目前devtools提供如下默认值:

properties.put("spring.thymeleaf.cache", "false"); //thymeleaf是个模板引擎
properties.put("spring.freemarker.cache", "false");
properties.put("spring.groovy.template.cache", "false");
properties.put("spring.velocity.cache", "false");
properties.put("spring.mustache.cache", "false"); //mustache也是个模板
properties.put("server.session.persistent", "true");
properties.put("spring.h2.console.enabled", "true"); //h2是个内存db
properties.put("spring.resources.cache-period", "0");



四、自动重启

开启devtools后,classpath中的文件变化会导致应用自动重启。Eclipse中保存文件即可引起classpath更新(注:需要打开自动编译),从而触发重启。 
如果不使用IDE,而是通过maven或者gradle的build插件来启动应用,也可以打开spring-boot-maven-plugin的fork配置,使得应用在单独进程中打开(这会使得应用使用自己的类加载器):

<build>
 <plugins>
  <!-- 添加spring-boot的maven插件 start-->
  <plugin>
  <groupId>org.springframework.boot</groupId>
  <artifactId>spring-boot-maven-plugin</artifactId>
  <configuration>
  <fork>true</fork>
  </configuration>
  </plugin>
  <!-- 添加spring-boot的maven插件 end-->
 </plugins>
</build>

restart与reload 
自动重启的原理在于spring boot使用两个classloader:不改变的类(如第三方jar)由base类加载器加载,正在开发的类由restart类加载器加载。应用重启时,restart类加载器被扔掉重建,而base类加载器不变,这样重启会很快。 
如果重启不够快,或者遇到类加载问题,可以使用reload技术,如JRebel或Spring Loaded等。



1. 排除resources

一些资源无需触发重启,例如thymeleaf模板文件就可以实时编辑。默认情况下,更改/META-INF/maven, /META-INF/resources ,/resources ,/static ,/public 或/templates下的资源不会触发重启,而是触发live reload(稍后会介绍)。 
可以使用spring.devtools.restart.exclude属性配置,例如 
spring.devtools.restart.exclude=static/**,public/** 
※ 如果想保留默认配置,同时增加新的配置,则可使用spring.devtools.restart.additional-exclude属性



2. 监视其他的path

不在classpath内的path可以配置spring.devtools.restart.additionalpaths属性来增加到监视中,同时配置spring.devtools.restart.exclude可以选择这些path的变化是导致restart还是live reload。



3. 关闭restart

① 在application.properties中配置spring.devtools.restart.enabled=false,此时restart类加载器还会初始化,但不会监视文件更新。 
② 在SprintApplication.run之前调用System.setProperty(“spring.devtools.restart.enabled”, “false”);可以完全关闭重启支持。



4. 使用触发文件

若不想每次修改都触发自动重启,可以设置spring.devtools.restart.trigger-file指向某个文件,只有更改这个文件时才触发自动重启。



5. 定制restart类加载器

默认时,IDE中打开的项目都会由restart加载器加载,jar文件由Base加载器加载,但是若你使用multi-module的项目,并且不是所有模块都被导入到IDE中,此时会导致加载器不一致。这时你可以创建META-INF/spring-devtools.properties文件,并增加restart.exclude.XXX,restart.include.XXX来配置哪些jar被restart加载,哪些被base加载。如:

restart.include.companycommonlibs=/mycorp-common-[\\w-]+\.jar
restart.include.projectcommon=/mycorp-myproj-[\\w-]+\.jar



6.自动重启演示

Hutool对应springBoot版本_开发工具



五、Live Reload

Devtools模块包含一个嵌入的LiveReload服务器,可以在资源变化时用来触发浏览器刷新。浏览器需要在livereload.com下载安装扩展。 
例如Chrome浏览器在应用商店安装LiveReload插件后,在要自动刷新的页面点击下图中图标,启动应用后更新页面内容或者css等都会触发页面自动刷新。

插件图标: 

Hutool对应springBoot版本_spring_02

示例1 - 修改css文件: 

Hutool对应springBoot版本_自动重启_03

示例2 - 修改java文件: 

Hutool对应springBoot版本_自动重启_04

若不想使用livereload,可以设置spring.devtools.livereload.enabled为false。 
※ 一次只能启动一个LiveReload,若在IDE上启动了多个应用,只有第一个有livereload支持。



六、全局设置

可以在$HOME路径下创建 .spring-boot-devtools.properties 文件,来包含所有使用了devtools的sprint boot应用都会使用的全局属性。例如,可以在文件中配置全局的trigger file:

~/.spring-boot-devtools.properties
spring.devtools.reload.trigger-file=.reloadtrigger