跳转到内容
主菜单
主菜单
移至侧栏
隐藏
导航
首页
最近更改
随机页面
MediaWiki帮助
代码酷
搜索
搜索
中文(中国大陆)
外观
创建账号
登录
个人工具
创建账号
登录
未登录编辑者的页面
了解详情
贡献
讨论
编辑“︁
死锁原理与预防
”︁(章节)
页面
讨论
大陆简体
阅读
编辑
编辑源代码
查看历史
工具
工具
移至侧栏
隐藏
操作
阅读
编辑
编辑源代码
查看历史
常规
链入页面
相关更改
特殊页面
页面信息
外观
移至侧栏
隐藏
您的更改会在有权核准的用户核准后向读者展示。
警告:
您没有登录。如果您进行任何编辑,您的IP地址会公开展示。如果您
登录
或
创建账号
,您的编辑会以您的用户名署名,此外还有其他益处。
反垃圾检查。
不要
加入这个!
= 死锁原理与预防 = == 定义与基本条件 == '''死锁'''(Deadlock)指两个或多个线程在执行过程中,因争夺资源而造成的一种互相等待的现象。若无外力干涉,这些线程将无法继续推进。死锁的发生必须同时满足以下四个必要条件(Coffman条件): # '''互斥条件''':资源一次只能被一个线程占用。 # '''占有并等待''':线程持有至少一个资源,并等待获取其他被占用的资源。 # '''非抢占条件''':已分配给线程的资源不能被其他线程强行夺取。 # '''循环等待条件''':存在一个线程的循环等待链,每个线程都在等待下一个线程所占用的资源。 数学描述(使用集合论表示): <math> \text{Deadlock} = \left\{ T_1, T_2, \ldots, T_n \mid T_i \rightarrow T_{i+1} \text{的等待关系形成环} \right\} </math> == 代码示例分析 == 以下Java代码展示了一个典型的死锁场景: <syntaxhighlight lang="java"> public class DeadlockExample { private static final Object lock1 = new Object(); private static final Object lock2 = new Object(); public static void main(String[] args) { Thread threadA = new Thread(() -> { synchronized (lock1) { System.out.println("ThreadA持有lock1"); try { Thread.sleep(100); } catch (InterruptedException e) {} synchronized (lock2) { System.out.println("ThreadA获取lock2"); } } }); Thread threadB = new Thread(() -> { synchronized (lock2) { System.out.println("ThreadB持有lock2"); synchronized (lock1) { System.out.println("ThreadB获取lock1"); } } }); threadA.start(); threadB.start(); } } </syntaxhighlight> '''输出结果可能为:''' <pre> ThreadA持有lock1 ThreadB持有lock2 (程序无限挂起)</pre> '''关键点说明:''' * 线程A持有`lock1`后尝试获取`lock2` * 线程B持有`lock2`后尝试获取`lock1` * 双方互相阻塞,形成循环等待 == 死锁检测与诊断 == === Java工具链 === 1. 使用`jstack`命令生成线程转储: <syntaxhighlight lang="bash"> jstack <pid> > thread_dump.txt </syntaxhighlight> 2. 查找输出中的`deadlock`关键词,典型报告如下: <pre> Found one Java-level deadlock: "ThreadB": waiting to lock monitor 0x00007f88a8003e58 (object 0x000000076ab270c8) which is held by "ThreadA" "ThreadA": waiting to lock monitor 0x00007f88a8003f98 (object 0x000000076ab270d8) which is held by "ThreadB"</pre> === 可视化分析 === <mermaid> graph LR A[ThreadA] -->|持有 lock1\n请求 lock2| B[ThreadB] B -->|持有 lock2\n请求 lock1| A </mermaid> == 预防策略 == {| class="wikitable" ! 策略 !! 实现方法 !! 示例 |- | '''破坏互斥条件''' || 使用共享资源(如读写锁) || `ReentrantReadWriteLock` |- | '''破坏占有并等待''' || 一次性申请所有资源 || <syntaxhighlight lang="java"> // 使用原子操作获取所有锁 while(!tryLockAll(lock1, lock2)) { releaseLocks(); backoff(); } </syntaxhighlight> |- | '''破坏非抢占条件''' || 设置超时机制 || <syntaxhighlight lang="java"> lock.tryLock(500, TimeUnit.MILLISECONDS) </syntaxhighlight> |- | '''破坏循环等待''' || 规定资源请求顺序 || <syntaxhighlight lang="java"> // 总是先申请lock1再申请lock2 </syntaxhighlight> |} == 实际案例:数据库事务死锁 == '''场景描述:''' * 事务T1先更新账户A再更新账户B * 事务T2先更新账户B再更新账户A '''解决方案:''' <syntaxhighlight lang="sql"> -- 规定所有事务必须按字母顺序操作账户 BEGIN TRANSACTION; UPDATE accounts SET balance = balance - 100 WHERE id = 'A'; UPDATE accounts SET balance = balance + 100 WHERE id = 'B'; COMMIT; </syntaxhighlight> == 高级话题:哲学家就餐问题 == 经典同步问题演示死锁: <mermaid> graph TB P1(哲学家1) -->|请求| F1(筷子1) P1 -->|请求| F2(筷子2) P2(哲学家2) -->|请求| F2 P2 -->|请求| F3(筷子3) P3(哲学家3) -->|请求| F3 P3 -->|请求| F4(筷子4) P4(哲学家4) -->|请求| F4 P4 -->|请求| F5(筷子5) P5(哲学家5) -->|请求| F5 P5 -->|请求| F1 </mermaid> '''解决方案:''' * 限制最多4位哲学家同时就餐 * 使用资源分级(筷子编号,总是先拿小号筷子) {{Warning|死锁预防会增加系统复杂度,需权衡安全性与性能}} == 总结 == * 死锁的四个必要条件缺一不可 * 预防策略围绕破坏必要条件展开 * 实际开发中应优先使用`java.util.concurrent`工具包 * 复杂系统建议结合静态分析和运行时检测 [[Category:计算机科学]] [[Category:面试技巧]] [[Category:Java并发编程]]
摘要:
请注意,所有对代码酷的贡献均被视为依照知识共享署名-非商业性使用-相同方式共享发表(详情请见
代码酷:著作权
)。如果您不希望您的文字作品被随意编辑和分发传播,请不要在此提交。
您同时也向我们承诺,您提交的内容为您自己所创作,或是复制自公共领域或类似自由来源。
未经许可,请勿提交受著作权保护的作品!
取消
编辑帮助
(在新窗口中打开)