SpringBoot热部署IDEA配置及原理
热部署
在开发项目过程中,当修改了某些代码后需要本地验证时,需要重启本地的服务进行验证,启动这个项目,如果项目庞大的话还是需要较长时间,spring开发团队为我们提供了一个插件:spring-boot-devtools,很好的解决了本地验证缓慢的问题。
热部署实现演示
添加spring-boot-devtools热部署依赖启动器
在Spring Boot 项目进行热部署测试之前,需要先在项目的pom.xml文件中添加spring-boot-devtools热部署依赖启动器:
<!-- 引入热部署依赖 -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-devtools</artifactId>
</dependency>
由于使用的是IDEA开发工具,添加热部署依赖后可能没有任何效果,接下来需要针对 IDEA开发工具进行部署相关功能设置
IDEA工具热部署设置
选择IDEA工具界面的File->Setting选项 ,打开Compiler面板设置页面
选择build下的Complier选项,在右侧勾选Build project automatically选项将 项目设置为自动编译,单机Apply ok按钮保存设置在项目任意页面中使用组合快捷键“Ctrl+Shift+Alt+/”打开Maintenance选项框,选中并打开Registry页面,具体如图1-17所示
列表中找到“compiler.automake.allow.when.app.running”,将该选项后的Value值勾选,用于指定IDEA工具在程序运行过程中自动编译,最后单击【Close】按钮完成设置
热部署原理分析
我们在编译器上启动项目,然后改动相关代码,然后编辑器自动触发编译替换掉 历史.class文件后,项目检测到文件变更后会重启spring-boot项目。可以看看官网的触发描述:
可以看到,我们引入插件后,,插件监控我们classpath的资源变化,当classpath有变化后,会出发重启
这里提到了,该插件重启快速的原因:这里对类加载采用了两种类加载器,对于第三方jar包采用baseclassloader来加载,对于开发人员自己开发的代码则使用restartClassLoader来进行加载,这使得比停掉服务重启要快的多,因为使用插件只是重启开发人员编写的代码部分。
这边做个简单的验证:
@Component
public class Devtools implements InitializingBean {
private static final Logger log = LoggerFactory.getLogger(Devtools.class);
@Override
public void afterPropertiesSet() throws Exception {
log.info("guava-jar classLoader: " +DispatcherServlet.class.getClassLoader().toString());
log.info("Devtools ClassLoader: " +
this.getClass().getClassLoader().toString());
}
}
这边先去除spring-boot-devtools插件,跑下工程:
可以看到,DispatcherServlet(第三方jar包)和Devtools(自己编写的类)使用的都是AppClassLoader加载的。我们现在加上插件,然后执行下代码:
发现第三方的jar包的类加载器确实是使用的系统的类加载器,而我们自己写的代码的类加载器为RestartClassLoader,并且每次重启,类加载器的实例都会改变。
排除资源
某些资源在更改后不一定需要触发重新启动。例如,Thymeleaf模板可以就地编辑。默认情况下,改变资源 /META-INF/maven , /META-INF/resources , /resources , /static , /public ,或 /templates 不触发重新启动,但确会触发现场重装。如果要自定义这些排除项,则可以使用spring.devtools.restart.exclude 属性。例如,仅排除 /static , /public 您将设置以下属性:
spring.devtools.restart.exclude=static/**,public/**