🏷️个人主页:牵着猫散步的鼠鼠
🏷️系列专栏:Java全栈-专栏
🏷️个人学习笔记,若有缺误,欢迎评论区指正
目录
前言
优雅停机是什么
SpringBoot如何实现优雅停机
合理杀死进程
法一:配置设置SpringBoot优雅停机
法二:配置Tomcat 容器优雅停机
设置Tomcat 容器关闭代码
设置 Tomcat 自动装配
总结
在服务器环境中,确保应用程序能够平滑关闭并处理完所有现有请求是一个重要的需求。Spring Boot 为我们提供了优雅退出的功能,使应用程序能够在关闭时正常处理完所有当前请求,避免请求被中断导致数据丢失或不一致等问题。本文将全面介绍如何在 Spring Boot 应用程序中实现优雅退出。
优雅停机(Graceful Shutdown) 是指在服务器需要关闭或重启时,能够先处理完当前正在进行的请求,然后再停止服务的操作。
优雅停机的实现步骤主要分为以下几步:
优雅停机的实现步骤分为以下两步:
在 Linux 中 kill 杀死进程的常用命令有以下这些:
因此,在以上命令中,我们不能使用“kill -9”来杀死进程,使用“kill”杀死进程即可。
在 Spring Boot 2.3.0 之后,可以通过配置设置开启 Spring Boot 的优雅停机功能,如下所示:
# 开启优雅停机,默认值:immediate 为立即关闭 server.shutdown=graceful # 设置缓冲期,最大等待时间,默认:30秒 spring.lifecycle.timeout-per-shutdown-phase=60s
此时,应用在关闭时,Web 服务器将不再接受新请求,并等待正在进行的请求完成的缓冲时间。
然而,如果是 Spring Boot 2.3.0 之前,就需要自行扩展(线程池)来实现优雅停机了。它的核心实现实现是在系统关闭时会调用 ShutdownHook,然后在 ShutdownHook 中阻塞 Web 容器的线程池,直到所有请求都处理完毕再关闭程序,这样就实现自定义优雅线下了。
但是,不同的 Web 容器(Tomcat、Jetty、Undertow)有不同的自定义优雅停机的方法,以 Tomcat 为例,它的自定义优雅停机实现如下。
实现 TomcatConnectorCustomizer 接口,用于在应用关闭时暂停 Tomcat 连接器:
public class TomcatGracefulShutdown implements TomcatConnectorCustomizer, ApplicationListener{ private volatile Connector connector; public void customize(Connector connector) { this.connector = connector; } public void onApplicationEvent(ContextClosedEvent contextClosedEvent) { this.connector.pause(); Executor executor = this.connector.getProtocolHandler().getExecutor(); if (executor instanceof ThreadPoolExecutor) { try { log.info("Start to shutdown tomcat thread pool"); ThreadPoolExecutor threadPoolExecutor = (ThreadPoolExecutor) executor; threadPoolExecutor.shutdown(); if (!threadPoolExecutor.awaitTermination(20, TimeUnit.SECONDS)) { log.warn("Tomcat thread pool did not shutdown gracefully within 20 seconds. "); } } catch (InterruptedException e) { log.warn("Fail to shut down tomcat thread pool ", e); } } } }
在配置类中声明该 Bean 并将其注册为 Tomcat 的 ConnectorCustomizer:
@Configuration @ConditionalOnClass({Servlet.class, Tomcat.class}) public static class TomcatConfiguration { @Bean public TomcatGracefulShutdown tomcatGracefulShutdown() { return new TomcatGracefulShutdown(); } @Bean public EmbeddedServletContainerFactory tomcatEmbeddedServletContainerFactory(TomcatGracefulShutdown gracefulShutdown) { TomcatEmbeddedServletContainerFactory tomcatFactory = new TomcatEmbeddedServletContainerFactory(); tomcatFactory.addConnectorCustomizers(gracefulShutdown); return tomcatFactory; } }
实现应用程序的优雅退出对于确保服务的可靠性和数据一致性至关重要。Spring Boot 从 2.3.0 版本开始提供了开箱即用的优雅退出支持,低于该版本则需要手动扩展实现。无论使用哪种方式,都应该遵循上述最佳实践,以确保应用程序能够安全、高效地关闭。同时,对于不同的 Web 容器,优雅退出的具体实现细节也会有所差异,需要合理配置。