如果您确实需要将构造函数的其余部分与以任何方式获取对尚未完全构造的对象的引用的任何线程进行同步,则可以使用同步块:
public class Test {
public Test() {
final Test me = this;
synchronized(this) {
new Thread() {
@Override
public void run() {
// ... Reference 'me,' the object being constructed
synchronized(me) {
// do something dangerous with 'me'.
}
}
}.start();
// do something dangerous with this
}
}
}
通常,像这样“提供”尚未构造的对象被认为是不好的样式,因此不需要同步构造函数。
在某些极端情况下,同步构造函数会很有用。这里有一个更现实的例子,来自Bozho答案的讨论:
public abstract class SuperClass {
public SuperClass() {
new Thread("evil") { public void run() {
doSomethingDangerous();
}}).start();
try {
Thread.sleep(5000);
}
catch(InterruptedException ex) { /* ignore */ }
}
public abstract void doSomethingDangerous();
}
public class SubClass extends SuperClass {
int number;
public SubClass () {
super();
number = 2;
}
public synchronized void doSomethingDangerous() {
if(number == 2) {
System.out.println("everything OK");
}
else {
System.out.println("we have a problem.");
}
}
}
我们希望只有在SubClass对象的构造完成后才调用该方法,例如,我们只需要“一切正常”的输出。但是在这种情况下,当您只能编辑子类时,您没有机会实现这一目标。如果构造函数可以同步,它将解决问题。doSomethingDangerous()
所以,我们从中学到的是:如果你的类不是最终的,永远不要像我在超类构造函数中所做的那样做一些事情 - 并且不要从你的构造函数中调用你自己类的任何非最终方法。