news 2026/7/2 12:18:46

线程池导致的 shutdown失败的完整排查过程

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
线程池导致的 shutdown失败的完整排查过程

SpringBoot 中有一种方式可以优雅地关闭应用程序。

(优雅停机是指关闭应用程序时,在规定的超时时间范围内,允许进行中的请求完成,拒绝新的请求进入。 这将使应用在请求处理方面保持一致,即没有未处理请求,每一个请求都被处理(完成或拒绝)

配置如下

yml server: port: 8888 shutdown: graceful management: endpoint: shutdown: enabled: true endpoints: web: exposure: include: shutdown
xml <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-actuator</artifactId> </dependency>

现象

直接调用 localhost:8888/actuator/shutdown 即可关闭应用程序。但是在调用某个业务后,再调用 shutdown 的 api,发现实际 shutdown 确实在执行,但是最终并没有把 pid 给 kill 掉,应用程序依然在运行。

第一怀疑就是认为这个程序执行后,还有什么资源没有被关闭掉,导致 springboot 认为应用程序还在运行,从而没有执行关闭操作。

排查过程

执行脚本

shell 生成线程快照 jstack -l pid > threads.txt # 查询非守护进程(因为非守护线程会阻止 JVM 退出) -v 表示反向排除 grep -n '" ' threads2.txt | grep -v daemon
  • 所有线程信息
  • 非守护线程信息

在里面发现了一段 关于 "pool-X-thread-Y"的线程信息,这个 ThreadPoolExecutor 出来的线程,处于等待中,其他的都是额外框架的线程信息或者 jvm 的,只有"pool-X-thread-Y"属于额外的。

less "pool-4-thread-1" #230 prio=5 os_prio=31 cpu=0.21ms elapsed=252.08s tid=0x00000001642cf800 nid=0x9a07 waiting on condition [0x000000017aaf6000] java.lang.Thread.State: WAITING (parking) at jdk.internal.misc.Unsafe.park(java.base@17.0.13/Native Method) - parking to wait for <0x0000000701e8af30> (a java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject) at java.util.concurrent.locks.LockSupport.park(java.base@17.0.13/LockSupport.java:341) at java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionNode.block(java.base@17.0.13/AbstractQueuedSynchronizer.java:506) at java.util.concurrent.ForkJoinPool.unmanagedBlock(java.base@17.0.13/ForkJoinPool.java:3465) at java.util.concurrent.ForkJoinPool.managedBlock(java.base@17.0.13/ForkJoinPool.java:3436) at java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject.await(java.base@17.0.13/AbstractQueuedSynchronizer.java:1630) at java.util.concurrent.ArrayBlockingQueue.take(java.base@17.0.13/ArrayBlockingQueue.java:420) at java.util.concurrent.ThreadPoolExecutor.getTask(java.base@17.0.13/ThreadPoolExecutor.java:1062) at java.util.concurrent.ThreadPoolExecutor.runWorker(java.base@17.0.13/ThreadPoolExecutor.java:1122) at java.util.concurrent.ThreadPoolExecutor$Worker.run(java.base@17.0.13/ThreadPoolExecutor.java:635) at java.lang.Thread.run(java.base@17.0.13/Thread.java:840) Locked ownable synchronizers: - None
  • ThreadPoolExecutor 默认线程名称源码

有了这个排查方向,去项目里面查找关于ThreadPoolExecutor的代码。

最终发现一句关于线程池的声明代码。从代码来看,虽然 XxxConfig 类上加了 @Configuration 注解,受到 spring 管理,但是 XXX_EXECUTOR 这个线程池是静态变量, 并没有受到 spring 管理,所以 springboot 在执行 shutdown 的时候,并不会关闭这个线程池,导致应用程序没有被关闭。

java @Configuration public class XxxConfig { public static final ThreadPoolExecutor XXX_EXECUTOR = new ThreadPoolExecutor(20, 20, 1000, TimeUnit.MILLISECONDS, new ArrayBlockingQueue<>(10000), new ThreadPoolExecutor.CallerRunsPolicy()); }

最终解决方案

建议将 XXX_EXECUTOR 这个线程池改为 spring 管理的 bean,如下所示:

java @Configuration public class XxxConfig { @Bean("xxxExecutor") public ThreadPoolExecutor xxxExecutor() { //示例 demo return new ThreadPoolExecutor(20, 20, 1000, TimeUnit.MILLISECONDS, new ArrayBlockingQueue<>(10000), new ThreadPoolExecutor.CallerRunsPolicy()); } }
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/7/2 1:13:45

Windows验机

跳过联网进入系统 检查是否有运输模式&#xff1a;不外接电源开不了机开机之后&#xff0c;不要操作&#xff0c;按 shiftf10 进入cmd输入 oobe\bypassnro &#xff0c;此时会重启并跳过联网进入系统 验机 查看电源通电时间&#xff1a;打开CMD终端&#xff0c;输入powercfg…

作者头像 李华
网站建设 2026/6/29 23:09:40

别让孩子视力提早“透支” ,这份护眼指南请收好

如今&#xff0c;电子产品成了孩子的“日常陪伴”&#xff0c;线上学习、娱乐样样离不开&#xff1b;叠加堆积如山的作业与课外辅导&#xff0c;双重压力下&#xff0c;越来越多孩子的视力早早亮起“红灯”——近视低龄化、高发化的趋势愈发严峻&#xff0c;不少家长刚上小学的…

作者头像 李华
网站建设 2026/7/2 10:55:59

儿童青少年近视干预科学指引,破解家长近视防控焦虑

近年来&#xff0c;我国经济社会快速发展推动电子产品全面普及&#xff0c;儿童青少年的生活与学习模式随之改变&#xff0c;眼健康问题愈发凸显。国家卫健委最新数据显示&#xff0c;我国儿童青少年总体近视率已达52.7%&#xff0c;且近视呈现明显的低龄化、重度化趋势&#x…

作者头像 李华
网站建设 2026/7/1 11:22:21

解析 .NET 核心基石:CTS、CLS 与 CLR 的核心价值与协同作用

.NET 框架的成功和其跨语言、跨平台能力的实现&#xff0c;离不开三大核心组件&#xff1a;通用类型系统 (CTS)、通用语言规范 (CLS) 和 公共语言运行库 (CLR)。这三者各自承担重要角色&#xff0c;但又紧密协作&#xff0c;共同构成了 .NET 生态的基础。掌握它们的作用是理解 …

作者头像 李华
网站建设 2026/6/28 18:26:58

Selinux权限的检测

Selinux权限的检测 当主体要真正去访问客体资源的时候&#xff0c;会去触发安全的认证的检测过程&#xff0c;比如说它有对应的权限&#xff0c;放行&#xff0c;如果没对应的权限&#xff0c;就访问失败&#xff0c;问题是&#xff0c;由谁来监控权限的合法性工作&#xff1f;…

作者头像 李华