这是《Java并发编程实战》3.2发布与逸出一节中的示例代码。我无法理解,this是怎么逸出的。
//隐式地使this引用逸出(不要这么做)
public class ThisEscape {
public ThisEscape(EventSource source){
source.registerListener(new EventListener() {
public void onEvent(Event e){
doSomething(e);
}
});
}
}
书上说,当ThisEscape发布EventListener时,也隐含地发布了ThisEscape实例本身,因为在这个内部类的实例中包含了对ThisEscape实例的隐含引用。
最近在看《Java 并发编程实战》,个人的理解:
首先,看里面的 doSomething(e) 方法,这个方法应该是在 ThisEscape 中,不然就无法解释。也就是说,通过 doSomething(e) 方法可以修改 ThisEscape 中的属性或者调用 ThisEscape 中的其他方法。
例子中的代码,在多线程环境下,会出现这样一种情况:
线程 A 和线程 B 同时访问 ThisEscape 构造方法,这时线程 A 访问构造方法还为完成(可以理解为 ThisEscape 为初始化完全),此时由于 this 逸出,导致 this 在 A 和 B 中都具有可见性,线程 B 就可以通过 this 访问 doSomething(e) 方法,导致修改 ThisEscape 的属性。也就是在 ThisEscape 还为初始化完成,就被其他线程读取,导致出现一些奇怪的现象。
这也就是 this 逸出。
通过 《Java 并发编程实战》 官网的书本 example 源码包,也证实了 doSomething 的确是 ThisEscape 中的方法。
Java Concurrency in Practice