线程死锁
死锁(DeadLock)
死锁概述
线程死锁是指两个或两个以上的线程互相持有对方所需要的资源,由于synchronized的特性,一个线程持有一个资源,或者说获得一个锁,在该线程释放这个锁之前,其它线程是获取不到这个锁的,而且会一直死等下去,因此这便造成了死锁。
死锁产生的条件
- 互斥条件:一个资源,或者说一个锁只能被一个线程所占用,当一个线程首先获取到这个锁之后,在该线程释放这个锁之前,其它线程均是无法获取到这个锁的。
- 占有且等待:一个线程已经获取到一个锁,再获取另一个锁的过程中,即使获取不到也不会释放已经获得的锁。
- 不可剥夺条件:任何一个线程都无法强制获取别的线程已经占有的锁
- 循环等待条件:线程A拿着线程B的锁,线程B拿着线程A的锁。。
死锁演示
package com.github.excellent01;
/**
* @auther plg
* @date 2019/5/18 12:12
*/
public class DeadLock implements Runnable {
private Object lock1 = new Object(); // 两个锁对象
private Object lock2 = new Object();
@Override
public void run() {
while(true){
method1();
method2();
}
}
public void method1(){
synchronized (lock1){
System.out.println(Thread.currentThread().getName() + "获取到lock1");
try {
Thread.sleep(1000); // 使得cpu切换
} catch (InterruptedException e) {
e.printstacktrace();
}
synchronized (lock2){
System.out.println(Thread.currentThread().getName() + "获取到lock2");
}
}
}
public void method2(){
synchronized (lock2){
System.out.println(Thread.currentThread().getName() + "获取到lock2");
synchronized (lock1){
System.out.println(Thread.currentThread().getName() + "获取到lock1");
}
}
}
public static void main(String[] args) {
DeadLock deadLock = new DeadLock();
new Thread(deadLock).start();
new Thread(deadLock).start();
}
}
运行结果:
这样便造成了死锁,Thread1拿着lock1,想要lock2,Thread2拿着lock2,想要lock1.
如何避免死锁
- 加锁顺序:线程按照相同的顺序加锁。
- 加锁时限,线程获取锁的过程中限制一定的时间,如果给定时间内获取不到,就算了,别勉强自己。这需要用到Lock的一些API。
死锁检测
在发生死锁之后,程序就卡住了没有任何反应,但程序仍在运行,因此需要借助一些
- 使用 jstack + id 进行跟踪排查
死锁的原因,具体哪个线程,具体的行号都说明的很清楚。
文章最后发布于: 2019-09-04 11:06:03
相关阅读
通过几个面试题来了解线程之间的通信。 问题清单: 如何让两个线程依次执行? 如何让两个线程按照指定方式有序交叉运行? 四个线程 A
进程和线程的区别 进程:是并发执行的程序在执行过程中分配和管理资源的基本单位,是一个动态概念,竞争计算机系统资源的基本单位。
https://github.com/jae-jae/camtdhttps://aria2.github.io/切换到解压目录D:\Tools\aria2-1.34.0-win-64bit中,打开cmd输入命令
PHP 从设计之初到流行起来都没有出现明显需要用多线程才能解决的需求。某些需要用到多线程的地方也有相应的解决方案和替代方案。
java-多线程-CountDownLatch(闭锁) CyclicBarrier(栅
(代码来源网络共享) 这几个工具类其实说白了就是为了能够更好控制线程之间的通讯问题~ CountDownLatch 是一个同步的辅助类,允许