public class ThisEscape {
public ThisEscape(EventResource source) {
source.registerListener(
new EventListener(){
public void onEvent(Event e) {
doSomething(e);
}
}
);
}
}
this引用在构造函数中逸出,当内部的EventListener发布时,外部的ThisEscape也逸出了(调用了doSomething)。对象构造函数还未返回,对象还不是完整的。所以如果this引用在构造过程中逸出,那么这种对象就会被认为是不正确的构造。
在构造过程中使用this引用逸出的一个常见错误是,在构造函数中启动一个线程。无论是显示创建还是隐式创建,在对象尚未构造完成之前,新的线程就可以看见它。在构造函数中创建线程并没有错误,但最好不要立即启动线程。在构造函数中调用一个可改写的实例方法时同样会导致this引用在构造过程中逸出。
如果想在构造函数中注册一个监听器或者启动线程,可以使用一个private的构造函数和一个public的静态工厂方法。
public class SafeListener {
private final EventListener listener;
private SafeListener() {
listener = new EventListener() {
public void onEvent(Event e) {
doSomething(e);
}
};
}
public static SafeListener newInstance(EventSource source) {
SafeListener safe = new SafeListener();
source.registerListener(safe.listener);
return safe;
}
}