以下示例摘自Brian Goetz第3章第3.5.1节的“实践中的Java并发性”一书。这是对象发布不当的示例
class someClass {
public Holder holder;
public void initialize() {
holder = new Holder(42);
}
}
public class Holder {
private int n;
public Holder(int n) { this.n = n; }
public void assertSanity() {
if (n!=n)
throw new AssertionError("This statement is false");
}
}
它说,持有人可能以不一致的状态出现在另一个线程上,而另一个线程可以观察到部分构造的对象。怎么会这样 您能否使用以上示例给出一个方案?
它还继续说,在某些情况下,线程在第一次读取字段时可能会看到过时的值,而在下次读取时可能会看到最新的值,这就是为什么assertSanity会引发Assertion
Error的原因。如何抛出assertionError?
从进一步的阅读来看,解决此问题的一种方法是通过使变量’n’为最终变量来使Holder不可变。现在,让我们假设Holder不能免疫,而实际上是不可变的。为了安全地发布此对象,我们是否必须使持有人初始化为静态并将其声明为volatile(静态初始化和volatile或只是volatile)?就像是
public class someClass {
public static volatile Holder holder = new Holder(42);
}
感谢您的帮助。