我正在学习Java的死锁,并且有来自Sun官方教程的以下示例代码:
阿方斯(Alphonse)和加斯顿(Gaston)是朋友,也是礼貌的忠实信徒。严格的礼貌规则是当您向朋友鞠躬时,您必须保持鞠躬,直到您的朋友有机会归还弓箭为止。不幸的是,该规则不能解决两个朋友可能同时鞠躬的可能性。
public class Deadlock {
static class Friend {
private final String name;
public Friend(String name) {
this.name = name;
}
public String getName() {
return this.name;
}
public synchronized void bow(Friend bower) {
System.out.format("%s: %s has bowed to me!%n",
this.name, bower.getName());
bower.bowBack(this);
}
public synchronized void bowBack(Friend bower) {
System.out.format("%s: %s has bowed back to me!%n",
this.name, bower.getName());
}
}
public static void main(String[] args) {
final Friend alphonse = new Friend("Alphonse");
final Friend gaston = new Friend("Gaston");
new Thread(new Runnable() {
public void run() { alphonse.bow(gaston); }
}).start();
new Thread(new Runnable() {
public void run() { gaston.bow(alphonse); }
}).start();
}
}
这是Sun的解释:
死锁运行时,两个线程极有可能在尝试调用bowBack时阻塞。两个块都不会结束,因为每个线程都在等待另一个退出弓。
我似乎不太了解。当alphonse.bow(gaston)运行时,bow方法被锁定。因此,现在它将首先打印“
Gaston向我鞠躬!”。然后它将继续并调用bowBack,并同时锁定bowBack。这怎么会造成僵局?我是否误解了调用同步方法时会发生什么?
如果有人可以给我一个简单的解释,谢谢。
需要注意的重要一点是,锁定的不是 方法 而是 对象实例 。
当您呼叫时alphonse.bow(gaston)
,它将尝试获取锁定alphonse
。拥有锁后,它将打印一条消息,然后调用gaston.bowBack(alphonse)
。此时,它尝试获取对的锁定gaston
。一旦拥有了锁,它将打印一条消息,然后释放该锁,最后释放该锁alphonse
。
在死锁中,锁的获取顺序使得任何一个线程都无法继续进行。
alphonse
gaston
gaston
-不能,因为线程2已经拥有它。alphonse
-不能,因为线程1已经拥有它。问题内容: 谁能解释一下为什么这段代码中会出现死锁。 问题答案: 这可能是如何执行的。 输入,由于关键字已锁定Alphonse 输入,加斯顿现已锁定 无法通过第一个方法调用执行,因为加斯顿(鲍尔)被锁定。等待锁被释放。 由于alphonse(上弦器)已锁定,因此无法从第二个方法调用执行。等待锁被释放。 两个线程都互相等待释放锁。
本文向大家介绍Oracle数据表中的死锁情况解决方法,包括了Oracle数据表中的死锁情况解决方法的使用技巧和注意事项,需要的朋友参考一下 在进行数据库管理的过程中,经常会出现数据表被用户的一些不合理操作而导致表被锁定的情况,以下主要介绍如何查找哪些表被哪个用户所锁定,以及如何解除锁定: 1.查找被锁定的表: 如果想知道具体是哪个进程阻塞了哪个进程,可用以下语句查看: 或 2.确定锁定表用户的si
本文向大家介绍请谈一谈,什么情况下会发生死锁?解决死锁的策略有哪些?相关面试题,主要包含被问及请谈一谈,什么情况下会发生死锁?解决死锁的策略有哪些?时的应答技巧和注意事项,需要的朋友参考一下 考察点:死锁 (一)互斥条件:一个资源一次只能被一个进程访问。即某个资源在一段时间内只能由一个进程占有,不能同时被两个或两个以上的进程占 有。这种独占资源如CD-ROM驱动器,打印机等等,必须在占有该资源的进
本文向大家介绍java 中死锁问题的实例详解,包括了java 中死锁问题的实例详解的使用技巧和注意事项,需要的朋友参考一下 java 中死锁问题的实例详解 先看代码在做解释 以上是代码部分,如果没有死锁,可以在if下加while(true),必然死锁,下面来做说明。 这个仅仅是为了理解死锁和面试用的,创建两个对象a和b只是为了作为死锁的对象而用,线程t1运行(t1.start()),线程t1拿到锁
本文向大家介绍详解Java中synchronized关键字的死锁和内存占用问题,包括了详解Java中synchronized关键字的死锁和内存占用问题的使用技巧和注意事项,需要的朋友参考一下 先看一段synchronized 的详解: synchronized 是 java语言的关键字,当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码。 一、当两个并发线程访
为什么java可重入锁不会导致死锁?