SpringBoot 优雅停机

在服务部署后,当需要停机重新发布时,我们会执行 kill -9 pid ,也行你已经把这些重启的命令写成了脚本,但是最终都是强制的kill 掉进程, 如果这是有人在访问接口,如果你突然将服务kill掉,是不是会造成客户的体验感降低,这时候就需要优雅的停机,何为优雅的停机呢?不难理解的是 如果有请求正在执行,这时我们需要进行kill的时候,需要处理完现在正在执行的请求,并且不再接受新的请求,最后处理完现有的请求进行停机服务; 如何做到这些么,其实网上有很多讲解,但是本文主要讲解的是SpringBoot2.3.0后带来的新特性

开启优雅停机服务

SpringBoot2.3.0后新增了如下特性,您只要将如下的代码添加到配置即可

server:
  shutdown: graceful

shutdown 提供了两种配置模式,默认是IMMEDIATE,源码如下:

public enum Shutdown {
	//优雅的停机服务
	GRACEFUL,
	//强制的关闭进程
	IMMEDIATE;
}

当然您还可以配置停机最大时间,默认30s

spring:
  # 优雅停机宽限时间
  lifecycle:
    timeout-per-shutdown-phase: 50

源码请点击这里arrow-up-right 配置完成,话不多说,让我们写个接口测试一下: 为了达到验证效果,这里小编,将方法进行了sleep

当我访问后,并立即关掉服务,这是你会发现日志如下:

可以发现我们的配置生效,并且也达到了,完成现有请求的处理,在进行服务停机 那么问题来了,我们为何配置了server.shutdown=graceful就可以实现我们想要的功能呢?

graceful配置的原理

在之前我们分析SpringBoot Tomcat启动原理时,有如下的方法: 如果您不了解SpringBoot Tomcat启动原理,请先阅读下面文章,会对您的了解有更多的帮助

  • getTomcatWebServer

这是我们启动时会自动读取配置,最终根据 shutdown的状态进行最终的执行;

但其实是由两种配置的,我们在回过头来看createWebServer()

其实这里已经帮我进行了两种方式的停机处理,但是会根据最终的 webServer 进行处理

我查看 webServer 会发现

而最终的处理就回到了根据

这里只讲 本文相关的优雅停机方式展现出来:

不得不说Spring的源码真的时一环扣一环,环环相扣,哈哈哈!

Last updated

Was this helpful?