0

0

在Java中如何定位和排查线程死锁_Java并发问题分析方法解析

P粉602998670

P粉602998670

发布时间:2026-02-04 06:05:23

|

165人浏览过

|

来源于php中文网

原创

jstack -l 可快速抓取线程快照并识别死锁,输出中出现“Found 1 deadlock.”即表明检测到循环等待导致的死锁,但仅覆盖synchronized和JUC显式锁,不包括StampedLock、外部资源锁等。

在java中如何定位和排查线程死锁_java并发问题分析方法解析

jstack 快速抓取线程快照并识别死锁

Java 自带的 jstack 是最直接、最低侵入的死锁定位工具。只要进程还在运行,就能立刻获取当前所有线程的状态和锁持有关系。

执行命令:

jstack -l 
其中 -l 参数关键——它会输出显式锁(ReentrantLock 等)的持有者和等待者,没有这个参数会漏掉大部分非 synchronized 死锁。

输出中一旦出现 Found 1 deadlock. 字样,后面会明确列出互相等待的线程、各自持有的锁和正在等待的锁。注意:这个提示只在 JVM 检测到「循环等待」时才打印,不是所有死锁都能被自动发现(比如涉及外部资源或自定义锁逻辑时)。

  • 确保用启动 Java 进程的同一用户执行 jstack,否则可能权限拒绝
  • 如果进程无响应但 jstack 报错「unable to open socket file」,可能是临时目录被清理,可尝试加 -J-Djava.io.tmpdir=/tmp
  • 线上环境慎用 jstack -F(强制模式),可能引起短暂 STW,尤其在 GC 压力大时

通过 ThreadMXBean 在代码中主动检测死锁

适合嵌入监控系统或测试阶段主动扫描,比人工 jstack 更可控。JDK 提供了 ThreadMXBean 接口,其 findDeadlockedThreads() 方法能返回当前 JVM 中所有陷入死锁的线程 ID 数组。

立即学习Java免费学习笔记(深入)”;

示例代码:

ThreadMXBean mxBean = ManagementFactory.getThreadMXBean();
long[] deadlockedIds = mxBean.findDeadlockedThreads();
if (deadlockedIds != null && deadlockedIds.length > 0) {
    ThreadInfo[] infos = mxBean.getThreadInfo(deadlockedIds, true, true);
    for (ThreadInfo info : infos) {
        System.err.println("Deadlocked thread: " + info.getThreadName());
        System.err.println("Stack trace:\n" + Arrays.toString(info.getStackTrace()));
    }
}

注意:findDeadlockedThreads() 只检测「synchronized 和 java.util.concurrent.Lock 的组合死锁」,不包含 StampedLockReadWriteLock 的读锁重入场景,也不覆盖 native 锁或数据库行锁等外部依赖。

AI封面生成器
AI封面生成器

专业的AI封面生成工具,支持小红书、公众号、小说、红包、视频封面等多种类型,一键生成高质量封面图片。

下载
  • 该方法是轻量级的,但频繁调用(如每秒多次)仍会带来一定开销
  • 返回 null 不代表绝对无死锁,只是 JVM 未检测到符合内置规则的循环等待
  • 配合 getThreadInfo(..., true, true) 才能拿到锁信息和完整栈,两个 true 缺一不可

使用 JMC 或 VisualVM 查看锁竞争热点

当死锁尚未发生,但已有明显线程阻塞或响应变慢时,需要观察锁的争用趋势。JMC(Java Mission Control)或 VisualVM 的 MBean 插件能实时显示 java.lang:type=Threading 下的锁统计。

重点关注三个指标:CurrentThreadCpuTime(是否某线程长期占用 CPU)、ThreadContentionMonitoringEnabled(需开启)、以及 PeakThreadCount 是否异常飙升。更实用的是「Sampler → Lock Instances」视图,它能列出被最多线程等待的锁实例(按 class+identity hash)。

  • JMC 默认不开启锁监控,需在启动 JVM 时加参数:-XX:+UnlockCommercialFeatures -XX:+FlightRecorder -XX:+FlightRecorderOptions=settings=profile
  • VisualVM 需安装「Threads» 插件,并勾选「Enable thread contention monitoring」
  • 看到某个 ReentrantLock 实例的 queueLength 持续 >5,且对应线程栈总卡在 lock(),大概率是锁粒度太粗或存在单点瓶颈

避免死锁的编码习惯比事后排查更重要

90% 的死锁源于可预测的模式:多把锁的获取顺序不一致、锁内调用外部方法、或在持有锁时等待条件变量。排查永远滞后于预防。

几个具体约束建议:

  • 所有涉及多把锁的操作,统一按「类名字母序 + 字段名」硬编码顺序获取,例如先 synchronized(A.class)synchronized(B.class),杜绝动态决定顺序
  • synchronized 块内禁止调用可能被子类重写的非 final 方法,防止隐式锁升级或回调死锁
  • tryLock(timeout, TimeUnit) 替代无超时的 lock(),并在超时后主动释放已持锁(需手动管理锁顺序)
  • ConcurrentHashMap 等线程安全容器,别误以为「不用锁」就绝对安全——它的迭代器弱一致性可能引发业务逻辑死锁(如遍历时修改触发 rehash,又在监听器里反向调用)

真正棘手的死锁往往藏在框架回调、代理增强、或跨 JVM 资源协调里,这时候 jstack 看不到锁,ThreadMXBean 也查不到——得结合日志埋点、分布式追踪 ID 和数据库锁视图交叉分析。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
什么是分布式
什么是分布式

分布式是一种计算和数据处理的方式,将计算任务或数据分散到多个计算机或节点中进行处理。本专题为大家提供分布式相关的文章、下载、课程内容,供大家免费下载体验。

373

2023.08.11

分布式和微服务的区别
分布式和微服务的区别

分布式和微服务的区别在定义和概念、设计思想、粒度和复杂性、服务边界和自治性、技术栈和部署方式等。本专题为大家提供分布式和微服务相关的文章、下载、课程内容,供大家免费下载体验。

236

2023.10.07

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

238

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

539

2024.03.01

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

238

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

539

2024.03.01

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1234

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

255

2025.10.17

全国统一发票查询平台入口合集
全国统一发票查询平台入口合集

本专题整合了全国统一发票查询入口地址合集,阅读专题下面的文章了解更多详细入口。

19

2026.02.03

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 3.2万人学习

C# 教程
C# 教程

共94课时 | 8.5万人学习

Java 教程
Java 教程

共578课时 | 57万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号