0

0

在Java中多线程调试有哪些技巧_Java并发问题排查思路说明

P粉602998670

P粉602998670

发布时间:2026-02-17 10:05:02

|

895人浏览过

|

来源于php中文网

原创

用jstack定位死锁需执行jstack -l ,关注末尾“found 1 deadlock”区块,明确列出互持/等待线程、锁地址及阻塞位置;注意权限与容器命名空间问题。

在java中多线程调试有哪些技巧_java并发问题排查思路说明

如何用jstack定位死锁线程

Java程序卡住、CPU不高但响应停滞,第一反应该看是否发生了死锁。jstack 是最轻量也最直接的工具,不需要改代码、不依赖IDE。

执行 jstack -l <pid></pid>-l 表示输出锁信息),重点关注输出末尾的 Found 1 deadlock. 区块。它会明确列出互相持有和等待的线程、锁对象地址、以及阻塞在哪个 synchronized 块或 Lock.lock() 调用上。

  • 注意:jstack 必须由与目标JVM相同用户运行,否则可能无权限;容器内需进到对应进程命名空间再执行
  • 如果没看到 Found deadlock,不代表没锁竞争——只是没构成循环等待,此时要结合 Thread.State: BLOCKED 线程堆栈,人工追踪锁对象(如 - waiting to lock )被谁持有
  • 使用 jstack -l <pid> > thread.log</pid> 保存快照,多次采样对比可发现“锁持有时间异常增长”的线索

为什么ThreadLocal变量在线程池中会引发脏数据

ThreadLocal 本身不是问题,问题出在复用线程时未清理。线程池中的线程长期存活,ThreadLocal 变量若没显式 remove(),下一次任务执行时可能读到上一个请求遗留的值。

典型表现是:HTTP请求间出现用户ID错乱、数据库连接配置污染、日志MDC上下文串号等。

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

EasySite
EasySite

零代码AI网站开发工具

下载
  • 必须在业务逻辑结束前调用 threadLocal.remove(),不能只靠 set(null) ——后者不释放底层 Entry,还可能导致内存泄漏
  • 在 Spring 环境中,优先用 @Scope("prototype") Bean 或 RequestContextHolder 替代手动管理 ThreadLocal
  • 若必须用,建议封装成工具类,在 try-finally 中强制 remove()
    try {
        userIdHolder.set(userId);
        doWork();
    } finally {
        userIdHolder.remove(); // 关键:不能省略
    }

如何用jcmd触发Java Flight Recorder(JFR)录制并发行为

当问题偶发、无法稳定复现,又需要观察线程状态切换、锁竞争、GC对并发的影响时,jcmd + JFR 比 jstack 更有力——它能持续采集数分钟内的底层事件,且开销可控(默认

先确认JVM启动时已启用JFR:-XX:+FlightRecorder(JDK8u262+ / JDK11+ 默认开启)。然后执行:

jcmd <pid> VM.unlock_commercial_features
jcmd <pid> JFR.start name=concurrent duration=120s settings=profile

录制结束后用 jcmd <pid> JFR.dump name=concurrent filename=recording.jfr</pid> 导出,用 JDK 自带的 JDK Mission Control 打开分析。

  • settings=profile 启用低开销采样,包含线程状态、锁持有时长、ForkJoinPool 任务队列深度等关键指标
  • 避免用 duration=0 长期录制,JFR 内存缓冲区有限,可能丢事件;建议按场景设定 30–180 秒窗口
  • 特别关注 “Monitor Blocked” 和 “Java Monitor Wait” 事件的频次与堆栈,比单纯看线程 dump 更容易定位热点锁

排查CompletableFuture链式调用丢失异常的常见盲点

CompletableFuture 的异步链中,若中间某个 thenApplythenAccept 抛出异常但没被 exceptionallyhandle 捕获,异常会被静默吞掉——主线程收不到,日志也不打,只剩任务“消失”。这是并发调试中最隐蔽的失败模式之一。

  • 所有非终端操作(thenRun, thenApply, thenCompose)都必须配对 exceptionally,或统一用 handle 处理结果与异常:
    future.thenApply(data -> process(data))
           .exceptionally(ex -> {
               log.error("process failed", ex);
               return fallbackValue;
           });
  • 不要依赖 join()get() 捕获异常——它们只能捕获最终阶段的异常,中间阶段失败会导致整个链提前终止且无提示
  • 在测试中主动注入异常(如用 completeExceptionally(new RuntimeException()))验证错误处理路径是否真实生效

实际并发问题往往不是单点故障,而是多个条件叠加:线程池饱和 + ThreadLocal未清理 + CompletableFuture异常静默 + 锁竞争加剧。调试时别急于加日志,先用 jstack 看住线程状态,用 jcmd + JFR 看清锁和任务流转,最后才深入代码逻辑。最容易被忽略的是——你以为的“异步完成”,其实早在线程池拒绝策略或 ForkJoinPool 工作窃取机制里就断掉了。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
spring框架介绍
spring框架介绍

本专题整合了spring框架相关内容,想了解更多详细内容,请阅读专题下面的文章。

142

2025.08.06

Java Spring Security 与认证授权
Java Spring Security 与认证授权

本专题系统讲解 Java Spring Security 框架在认证与授权中的应用,涵盖用户身份验证、权限控制、JWT与OAuth2实现、跨站请求伪造(CSRF)防护、会话管理与安全漏洞防范。通过实际项目案例,帮助学习者掌握如何 使用 Spring Security 实现高安全性认证与授权机制,提升 Web 应用的安全性与用户数据保护。

81

2026.01.26

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

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

244

2023.09.22

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

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

746

2024.03.01

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

418

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

593

2023.08.10

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

418

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

593

2023.08.10

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

462

2026.02.13

热门下载

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

精品课程

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

共23课时 | 3.6万人学习

C# 教程
C# 教程

共94课时 | 9.6万人学习

Java 教程
Java 教程

共578课时 | 67.2万人学习

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

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