3 回答
TA贡献1868条经验 获得超4个赞
双重检查锁定已损坏。由于初始化是原语,因此它可能不需要易失性即可工作,但是在实例初始化之前,没有什么可以阻止初始化被视为对非同步代码是正确的。
编辑:为了澄清上述答案,原始问题询问有关使用布尔值来控制双重检查锁定的问题。如果没有以上链接中的解决方案,它将无法正常工作。您可以仔细检查锁,实际上是设置一个布尔值,但是在创建类实例时,仍然存在有关指令重新排序的问题。建议的解决方案不起作用,因为在非同步块中看到已初始化的布尔值为true后,实例可能未初始化。
仔细检查锁定的正确解决方案是使用volatile(在实例字段上)并忽略初始化的布尔值,并确保使用JDK 1.5或更高版本,或在最终字段中对其进行初始化(如链接中所述)文章和汤姆的答案,或者只是不使用它。
当然,整个概念似乎是一个巨大的过早优化,除非您知道在获取此Singleton时将引起大量线程争用,或者您已剖析了该应用程序并将其视为热点。
TA贡献1998条经验 获得超6个赞
如果initialized是这样的话会奏效volatile。就像synchronized的有趣影响volatile一样,与参考数据的关系并不像我们对其他数据的说法那么重要。在写入之前,必须进行instance字段和Test对象的设置。当通过短路使用缓存的值时,读取发生在读取和通过引用到达的对象之前。拥有一个单独的标志并没有明显的区别(除了它会导致代码更加复杂)。initializedinitializeinstanceinitialized
(final不安全发布的构造函数中的字段规则有些不同。)
但是,在这种情况下,您应该很少看到该错误。首次使用时遇到麻烦的机会很小,这是一次非重复性的比赛。
代码过于复杂。您可以将其编写为:
private static final Test instance = new Test();
public static Test getInstance() {
return instance;
}
TA贡献1860条经验 获得超8个赞
双重检查的锁定确实被破坏了,并且解决该问题的方法实际上比该惯用语更容易实现代码方式-只需使用静态初始化程序即可。
public class Test {
private static final Test instance = createInstance();
private static Test createInstance() {
// construction logic goes here...
return new Test();
}
public static Test getInstance() {
return instance;
}
}
可以确保在JVM首次加载类时以及在类引用可以返回到任何线程之前执行静态初始化程序,从而使其固有地具有线程安全性。
添加回答
举报