《Java 并发编程实战》
这是一个之前的问题,那时候查的各种博客解决的
提供参考的博客:博客
在第二章线程安全 重入这一小节有这样的一段话:
“然而,如果内置锁是不可重入的。那么在调用super.dosomething时将无法获得Widget上的锁,因为这个锁已经被持有。”
public class Widget {
public synchronized void doSomething() {
...
}
}
public class LoggingWidget extends Widget {
public synchronized void doSomething() {
System.out.println(toString() + ": calling doSomething");
super.doSomething();
}
}
可能很多小伙伴看到这句话及其懵逼,懵逼书下的你我他。然后在不断查资料,学习就是含着辛酸泪啊!
直入正题。
原文:“因为这个锁已经被持有”,那为何会被持有。
首先我们需要知道此时super和this的关系。
现在咱们就先来看下下面的这个例子:
public class Widget {
public synchronized void doSomething() {
// ……
System.out.println("类Widget -> this: " + this);
}
}
public class LoggingWidget extends Widget {
public synchronized void doSomething() {
// System.out.println(toString() + ": calling doSomething");
super.doSomething();
System.out.println("类LoggingWidget -> this: " + this);
System.out.println("类LoggingWidget -> super: " + super.toString());
}
}
public class Super {
public static void main(String[] args) {
LoggingWidget lWidget = new LoggingWidget();
lWidget.doSomething();
}
}
运行结果:
类Widget -> this: panda.ceshi.java.LoggingWidget@15db9742
类LoggingWidget -> this: panda.ceshi.java.LoggingWidget@15db9742
类LoggingWidget -> super: panda.ceshi.java.LoggingWidget@15db9742
从中我们可以清楚的看见类LoggingWidget中的当前对象this、super和和父类中的this返回的都是同一对象的哈希值。也可以说LoggingWidget中的super其实就是和该类的this有着同样的引用。
当在创建子类对象时,子类会隐式或显示地调用父类构造方法,并继承父类的中的成员。这是因为在构造子类对象时,对应内存中会分别存有来自该类的数据、来自父类的数据,但是他们都是由同一对象来访问,也就是该子类对象。这时候则通过关键字this访问来自该类的数据,通过super访问来自父类的数据。
然后我们在回顾一下什么叫重入:
当某个线程请求一个由其它线程持有的对象锁时,发出请求的线程就会阻塞,而当线程请求由自己持有的对象锁时,那么这个请求就会成功。即当一个线程获取对象锁之后,这个线程可以再次获取该对象上的锁。相对应这样的锁(ReentrantLock 和synchronized )就可重入。
所以在上面例子执行 lWidget.doSomething() 时,会先获取 lWidget对象锁(方法锁),当调用 super.doSomething() 时,则仍是去请求该对象锁,虽然该锁已经被持有,但synchronized是可重入的,所以请求成功。
最终解决 “如果内置锁是不可重入的。那么在调用super.dosomething时将无法获得Widget上的锁,因为这个锁已经被持有。” 疑惑。