Java死锁

一、什么是死锁?

死锁是并发编程中最隐蔽、最危险的问题之一,指两个或多个线程在执行过程中,因互相持有对方需要的资源而陷入无限等待的状态,若无外力干涉,所有线程将永远阻塞,程序无法继续运行。

简单来说:你拿着我要的锁,我拿着你要的锁,谁都不放手,谁都动不了。

二、死锁产生的 4 个必要条件

死锁并非随机出现,必须同时满足以下 4 个条件才会发生,破坏任意一个即可避免死锁:

  1. 互斥条件:资源同一时间只能被一个线程持有,其他线程必须等待。
  2. 请求与保持条件:线程已经持有至少一个资源,又去请求其他资源,且不释放已持有的资源。
  3. 不可剥夺条件:线程持有的资源只能由自己主动释放,不能被其他线程强行抢占。
  4. 循环等待条件:多个线程之间形成首尾相接的循环等待资源关系,如 A 等 B、B 等 C、C 等 A。

三、死锁经典场景与代码示例

最常见的死锁场景是线程交叉获取多把锁,以下是极简可复现代码:

复制代码
public class DeadLockDemo {
    private static final Object lockA = new Object();
    private static final Object lockB = new Object();

    public static void main(String[] args) {
        // 线程1:先拿lockA,再拿lockB
        new Thread(() -> {
            synchronized (lockA) {
                System.out.println("线程1获取lockA");
                try { Thread.sleep(100); } catch (InterruptedException e) {}
                synchronized (lockB) {
                    System.out.println("线程1获取lockB");
                }
            }
        }).start();

        // 线程2:先拿lockB,再拿lockA
        new Thread(() -> {
            synchronized (lockB) {
                System.out.println("线程2获取lockB");
                try { Thread.sleep(100); } catch (InterruptedException e) {}
                synchronized (lockA) {
                    System.out.println("线程2获取lockA");
                }
            }
        }).start();
    }
}

运行后,两个线程会分别持有一把锁,互相等待对方释放锁,触发死锁。

四、如何检测死锁?

实际项目中死锁不会直接报错,需要通过工具定位:

  1. jstack 命令 执行jps找到进程 ID,再用jstack -l 进程ID查看线程堆栈,日志中会明确提示Found one Java-level deadlock,并列出死锁线程与锁信息。
  2. JConsole/JVisualVM 打开可视化工具,连接进程后查看线程 面板,点击检测死锁,可直观看到死锁的线程与资源依赖关系。
  3. Arthas 阿里开源诊断工具,使用thread -b命令可直接定位阻塞的死锁线程。

五、死锁的避免与解决方法

核心思路:破坏死锁 4 个必要条件中的任意一个,推荐以下实用方案:

  1. 固定锁获取顺序 所有线程按相同顺序获取锁,避免循环等待,这是最常用、最安全的方案。
  2. 避免同时持有多把锁重构代码,让每个线程只持有一把锁,减少锁嵌套,从源头降低死锁概率。
  3. 使用定时锁Lock.tryLock(long time, TimeUnit unit)替代synchronized,获取锁超时则放弃并释放已持有的锁。
  4. 响应中断 使用lockInterruptibly()支持锁中断,等待锁时可被外部中断,避免无限等待。
  5. 减少锁粒度拆分大锁为多个小锁,降低线程竞争概率,提升并发效率。

六、总结

死锁是并发编程的经典陷阱,看似简单却极易在复杂业务中隐藏。理解 4 个必要条件、掌握检测工具、遵守编码规范,就能有效避免死锁。

开发中牢记:少用锁嵌套、固定加锁顺序、优先使用定时锁,让你的并发程序稳定运行,远离死锁困扰。

相关推荐
暮冬-  Gentle°2 小时前
自定义内存检测工具
开发语言·c++·算法
娇娇yyyyyy2 小时前
QT编程(15): Qt 按键事件和定时器事件
开发语言·qt
2501_945424802 小时前
C++编译期矩阵运算
开发语言·c++·算法
yy我不解释2 小时前
关于comfyui的mmaudio音频生成插件时时间不一致问题(三)
开发语言·python·ai作画·音视频·comfyui
2301_815482932 小时前
C++中的类型标签分发
开发语言·c++·算法
SuperEugene2 小时前
Vue3 模板语法规范实战:v-if/v-for 不混用 + 表达式精简,避坑指南|Vue 组件与模板规范篇
开发语言·前端·javascript·vue.js·前端框架
xushichao19892 小时前
代码生成优化技术
开发语言·c++·算法
leaves falling2 小时前
C++类和对象(1)
开发语言·c++
2401_873204652 小时前
模板编译期循环展开
开发语言·c++·算法