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 启动时会运行两次

如果我们重新启动 Spring Boot 应用程序,正在运行的用户线程发生了啥?

spring boot 自定义验证器运行两次

Apache Ignite 使用 Spring-Boot 加载两次?

使用 Spring Boot“bootRun”启动的应用程序在包含 Gradle Vaadin 插件时会导致 NoClassDefFoundError

停止 spring boot 应用程序启动,直到 spring cloud config server 启动