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 个必要条件、掌握检测工具、遵守编码规范,就能有效避免死锁。

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

相关推荐
FQNmxDG4S8 小时前
Java多线程编程:Thread与Runnable的并发控制
java·开发语言
前端老石人8 小时前
HTML 字符引用完全指南
开发语言·前端·html
matlab_xiaowang9 小时前
Redux 入门:JavaScript 可预测状态管理库
开发语言·javascript·其他·ecmascript
虹科网络安全9 小时前
艾体宝干货|数据复制详解:类型、原理与适用场景
java·开发语言·数据库
axng pmje9 小时前
Java语法进阶
java·开发语言·jvm
rKWP8gKv79 小时前
Java微服务性能监控:Prometheus与Grafana集成方案
java·微服务·prometheus
老前端的功夫10 小时前
【Java从入门到入土】28:Stream API:告别for循环的新时代
java·开发语言·python
qq_4352879210 小时前
第9章 夸父逐日与后羿射日:死循环与进程终止?十个太阳同时值班的并行冲突
java·开发语言·git·死循环·进程终止·并行冲突·夸父逐日
小江的记录本10 小时前
【Kafka核心】架构模型:Producer、Broker、Consumer、Consumer Group、Topic、Partition、Replica
java·数据库·分布式·后端·搜索引擎·架构·kafka
止语Lab10 小时前
从手动到框架:Go DI 演进的三个拐点
开发语言·后端·golang