spring boot 启动时会运行两次
Posted 鬼王呵
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了spring boot 启动时会运行两次相关的知识,希望对你有一定的参考价值。
1、背景
搭建的spring boot工程,在启动类中开始行加入一个输出(如下代码),并执行,发现这个输出执行了两次(打印了两次“tianji”)
public static void main(String[] args)
System.out.println("tianji");
SpringApplication.run(TacoCloudApplication.class, args);
为了一探究竟,尝试在在启动类的最后行加入一个输出(如下代码),并执行,发现后面加入的输出只执行了一次(打印了一次“hsx”)
public static void main(String[] args)
System.out.println("tianji");
SpringApplication.run(TacoCloudApplication.class, args);
System.out.println("hsx");
执行的结果输出
2、为什么会打印两次?
罪魁祸首是pom文件引入了热部署的二方包引起的,将其删除或者注释,就不会出现上面的问题
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-devtools</artifactId>
<scope>runtime</scope>
<optional>true</optional>
</dependency>
3、测试
将上述讲解的pom中的二方包注释,重新执行,发现每个输出都只会执行一次
4、为什么需要热部署&及其原理
4.1、热部署
spring为开发者提供了一个名为spring-boot-devtools的模块来使Spring Boot应用支持热部署,提高开发者的开发效率,无需手动重启Spring Boot应用
4.2、devtools的原理
深层原理是使用了两个ClassLoader,一个Classloader加载那些不会改变的类(第三方Jar包),另一个ClassLoader加载会更改的类,称为restart ClassLoader,这样在有代码更改的时候,原来的restart ClassLoader 被丢弃,重新创建一个restart ClassLoader,由于需要加载的类相比较少,所以实现了较快的重启时间
4.3、说明
4.3.1、特别说明
(1)devtools可以实现页面热部署(即页面修改后会立即生效,这个可以直接在application.properties文件中配置spring.thymeleaf.cache=false来实现),
实现类文件热部署(类文件修改后不会立即生效),实现对属性文件的热部署。
即devtools会监听classpath下的文件变动,并且会立即重启应用(发生在保存时机),注意:因为其采用的虚拟机机制,该项重启是很快的
(2)配置了后在修改java文件后也就支持了热启动,不过这种方式是属于项目重启(速度比较快的项目重启),会清空session中的值,也就是如果有用户登陆的话,项目重启后需要重新登陆。默认情况下,/META-INF/maven,/META-INF/resources,/resources,/static,/templates,/public这些文件夹下的文件修改不会使应用重启,但是会重新加载(devtools内嵌了一个LiveReload server,当资源发生改变时,浏览器刷新)
4.3.2、devtools的配置
(1)在application.properties中配置spring.devtools.restart.enabled=false,此时restart类加载器还会初始化,但不会监视文件更新
(2)在SprintApplication.run之前调用System.setProperty(“spring.devtools.restart.enabled”, “false”);可以完全关闭重启支持,配置内容:
#热部署生效
spring.devtools.restart.enabled: true
#设置重启的目录
#spring.devtools.restart.additional-paths: src/main/java
#classpath目录下的WEB-INF文件夹内容修改不重启
spring.devtools.restart.exclude: WEB-INF/**
4.3.3、IDEA配置
当我们修改了Java类后,IDEA默认是不自动编译的,而spring-boot-devtools又是监测classpath下的文件发生变化才会重启应用,所以需要设置IDEA的自动编译(MAC):
(1)Intellij IDAE-Preferences-Settings-Compiler-Build Project automatically
(2)option + shift + command + /,选择Registry,勾上 Compiler autoMake allow when app running (这个在ideaIU-2021.2没有找到)
以上是关于spring boot 启动时会运行两次的主要内容,如果未能解决你的问题,请参考以下文章
如果我们重新启动 Spring Boot 应用程序,正在运行的用户线程发生了啥?
Apache Ignite 使用 Spring-Boot 加载两次?
使用 Spring Boot“bootRun”启动的应用程序在包含 Gradle Vaadin 插件时会导致 NoClassDefFoundError