1,什么是死锁
死锁最初由一个悲惨的故事说起,话说一群哲学家一起聚餐,然后在每个人的左边和右边分别放着一根筷子,而只有同时抓到两根筷子,才能正常吃饭,于是,不幸的故事发生了,每位哲学家都只抓到一根筷子,且都不愿意释放手中的筷子,于是,最终一桌的饭菜就这么浪费了。
不知道这个故事是谁发明的,但确实形象说明了死锁的情况。
转换到线程的场景,就是线程A持有独占锁资源a,并尝试去获取独占锁资源b
同时,线程B持有独占锁资源b,并尝试去获取独占锁资源a
这样线程A和线程B相互持有对方需要的锁,从而发生阻塞,最终变为死锁。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40
| public class Deadlock {
private static final Object a = new Object(); private static final Object b = new Object();
public static void main(String[] args){ new Thread(new Task(true)).start(); new Thread(new Task(false)).start(); }
static class Task implements Runnable{ private boolean flag;
public Task(boolean flag){ this.flag = flag; }
@Override public void run() { if(flag){ synchronized (a){ System.out.println(Thread.currentThread().getName()+"->获取到a资源"); synchronized (b){ System.out.println(Thread.currentThread().getName()+"->获取到b资源"); } } }else{ synchronized (b){ System.out.println(Thread.currentThread().getName()+"->获取到b资源"); synchronized (a){ System.out.println(Thread.currentThread().getName()+"->获取到a资源"); } } }
} } }
|
2,如何防止死锁?(重点)
2.1 减少同步代码块嵌套操作
2.2 降低锁的使用粒度,不要几个功能共用一把锁
2.3 尽量采用tryLock(timeout)的方法,可以设置超时时间,这样超时之后,就可以主动退出,防止死锁(关键)