- 插件会监控 classpath的资源变化
- 对类加载采用了两种类加载器(只重启开发人员编写的代码部分)
一、 现象
开发项目过程中,修改了某些代码后需要本地验证时,需要重启本地服务进行验证。
如果项目庞大的话是需要较长时间的,Spring开发团队为我们带来了一个插件:spring-boot-devtools,很好的解决了本地验证缓慢的问题。
-
1.添加spring-boot-devtools热部署依赖启动器
在Spring Boot项目进行热部署测试之前,需要先在项目的pom.xml文件中添加spring-boot-devtools热部署依赖启动器:org.springframework.boot spring-boot-devtools
-
IDEA工具热部署设置(针对IDEA开发工具进行热部署相关的功能设置 )
选择IDEA工具界面的- 【File】
- 【Settings】选项
- 打开Compiler面板设置页面
- 选择Build下的Compiler选项,在右侧勾选“Build project automatically”选项将项目设置为自动编译,单击【Apply】→【OK】按钮保存设置
idea中开启自动编译
-
指定IDEA工具在程序运行过程中自动编译
- 在项目任意页面中使用组合快捷键“Ctrl+Shift+Alt+/”
- 打开Maintenance选项框,选中并打开Registry页面
- 列表中找到“compiler.automake.allow.when.app.running”
- 将该选项后的Value值勾选,用于指定IDEA工具在程序运行过程中自动编译
- 最后单击【Close】按钮完成设置
- 启动SpringBoot项目,访问controller
http://localhost:8080/test
- 清空浏览器控制台
- 修改controller层返回结果
- 刷新浏览器,已经查看到新的返回数据,表示热部署成功(未重启项目)
- 查看idea控制台日志
- 在编辑器上启动项目
- 然后改动相关的代码
- 此时编辑器自动触发编译替换掉历史的.class文件
- 项目检测到有文件变更后会重启SpringBoot项目
官网的触发描述:
引入了插件后,插件会监控classpath的资源变化,当classpath有变化后,会触发重启
- 对类加载采用了两种类加载器,对于第三方jar包采用base-classloader来加载,对于开发人员自己开发的代码则使用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。
并且每次重启,类加载器的实例都会如下图一样的改变



