【java】java如何理解隐式地使this引用逸出
这是《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 中的方法。
package net.jcip.examples;/**
* ThisEscape
* <p/>
* Implicitly allowing the this reference to escape
*
* @author Brian Goetz and Tim Peierls
*/
public class ThisEscape {
public ThisEscape(EventSource source) {
source.registerListener(new EventListener() {
public void onEvent(Event e) {
doSomething(e);
}
});
}
void doSomething(Event e) {
}
interface EventSource {
void registerListener(EventListener e);
}
interface EventListener {
void onEvent(Event e);
}
interface Event {
}
}
实话实说多线程、逸出我不是很懂,但是我懂内部类,所以可以来强答一下,内部类、匿名内部类都可以访问外部类的对象的域,为什么会这样,实际上是因为内部类构造的时候,会把外部类的对象this隐式的作为一个参数传递给内部类的构造方法,这个工作是编译器做的,他会给你内部类所有的构造方法添加这个参数,所以你例子里的匿名内部类在你构造ThisEscape时就把ThisEscape创建的对象隐式的传给匿名内部类了。至于这样会出什么问题我并不明白,貌似是怕onEvent会操作外部类的私有域?这部分需要你来教我了
读了楼上的答案,特别是2楼的答案就想明白了。接2楼,匿名内部类持有外部类的this引用后可能会调用外部类的实例方法(非私有方法,也不是final方法),这个时候当然是通过this引用调用的啦(代码中没有写this.doSomething(e),这是一种简写)。这个使用this引用指向的对象在还没有完全构造完就调用其实例方法改变属性,可能出现问题。比如外部类初始化的时候对一个int类型的参数赋值,然后doSomething(e)方法对该参数进行加1操作。如果onEvent()方法触发在外部类完成初始化之前怎么办?就会很尴尬。因为this引用提前被EventListener实例对象拿到。这个提前的拿到就是this引用的逸出。
所以书上最后总结说,在构造函数中调用可以可改写的实例方法(不是私有方法,也不是final方法),同样会导致this引用在构造过程中逸出。还有构造函数中不能启动线程也是一样,即在没完成初始化时私自调用对象的方法,改变其状态,会存在隐患。
并且书中又说,解决方法是,在构造函数中注册一个事件监听器或者启动线程,可以使用一个私有的构造函数和一个公共的工厂方法(Factory Method),从而避免不正确的构造过程。如图
这样就使得注册监听在构造之后执行,保证onEvent()方法被调用在SafeListener的构造之后,对象正确初始化后再调用this引用指向的对象的方法修改属性就不是逸出,而是发布。
以上是 【java】java如何理解隐式地使this引用逸出 的全部内容, 来源链接: utcz.com/a/73649.html