5

为什么下面的代码不会导致死锁?我的意思是在我调用 getNumber(.) 之后,Test 类的对象应该被锁定,所以我不应该能够访问 getNumber2(.)。

class Test() {
    synchronized int getNumber(int i){
        return getNumber2(i);
    }

    synchronized int getNumber2(int i) {
        return i;
    }

    public static void main(String[] args) {
        System.out.println((new Test()).getNumber(100));
    }
}

输出:

100
4

2 回答 2

20

这是因为锁是可重入的,这意味着它可以被同一个线程多次获取。

来自Java 教程

可重入同步

回想一下,一个线程不能获得另一个线程拥有的锁。但是线程可以获取它已经拥有的锁。允许一个线程多次获取同一个锁可以实现重入同步。这描述了一种情况,同步代码直接或间接调用一个也包含同步代码的方法,并且两组代码都使用相同的锁。如果没有可重入同步,同步代码将不得不采取许多额外的预防措施来避免线程导致自身阻塞。

JLS 的相关部分是§17.1。同步

Java 编程语言为线程之间的通信提供了多种机制。这些方法中最基本的是同步,它是使用监视器实现的。Java 中的每个对象都与一个监视器相关联,线程可以锁定或解锁监视器。一次只有一个线程可以锁定监视器。任何其他试图锁定该监视器的线程都会被阻塞,直到它们能够获得对该监视器的锁定。一个线程 t 可能会多次锁定一个特定的监视器;每次解锁都会反转一次锁定操作的效果。

于 2013-03-04T14:35:41.137 回答
4

它不会导致死锁,因为当一个线程进入一个同步方法时,它所做的就是检查它是否有一个锁this,如果没有,它会等到它可以拥有锁并获得它。

在您的情况下,当线程进入第二个同步方法时,它已经拥有this对象上的锁,因此它可以在不阻塞的情况下进入该方法。

于 2013-03-04T14:37:34.180 回答