Java中的双重检查(Double-Check)详解
在 Effecitve Java 一书的第 48 条中提到了双重检查模式,并指出这种模式在 Java 中通常并不适用。该模式的结构如下所示:
public Resource getResource() {
if (resource == null) {
synchronized(this){
if (resource==null) {
resource = new Resource();
}
}
}
return resource;
}
该模式是对下面的代码改进:
public synchronized Resource getResource(){
if (resource == null){
resource = new Resource();
}
return resource;
}
这段代码的目的是对 resource 延迟初始化。但是每次访问的时候都需要同步。为了减少同步的开销,于是有了双重检查模式。
在 Java 中双重检查模式无效的原因是在不同步的情况下引用类型不是线程安全的。对于除了 long 和 double 的基本类型,双重检查模式是适用 的。比如下面这段代码就是正确的:
private int count;
public int getCount(){
if (count == 0){
synchronized(this){
if (count == 0){
count = computeCount(); //一个耗时的计算
}
}
}
return count;
}
上面就是关于java中双重检查模式(double-check idiom)的一般结论。但是事情还没有结束,因为java的内存模式也在改进中。Doug Lea 在他的文章中写道:“根据最新的 JSR133 的 Java 内存模型,如果将引用类型声明为 volatile,双重检查模式就可以工作了”。所以以后要在 Java 中使用双重检查模式,可以使用下面的代码:
private volatile Resource resource;
public Resource getResource(){
if (resource == null){
synchronized(this){
if (resource==null){
resource = new Resource();
}
}
}
return resource;
}
当然了,得是在遵循 JSR133 规范的 Java 中。
所以,double-check 在 J2SE 1.4 或早期版本在多线程或者 JVM 调优时由于 out-of-order writes,是不可用的。 这个问题在 J2SE 5.0 中已经被修复,可以使用 volatile 关键字来保证多线程下的单例。
public class Singleton {
private volatile Singleton instance = null;
public Singleton getInstance() {
if (instance == null) {
synchronized(this) {
if (instance == null) {
instance = new Singleton();
}
}
}
return instance;
}
}
推荐方法 是Initialization on Demand Holder(IODH),
public class Singleton {
static class SingletonHolder {
static Singleton instance = new Singleton();
}
public static Singleton getInstance(){
return SingletonHolder.instance;
}
}
以上就是本文的全部内容,希望对大家学习java程序设计有所帮助。相关阅读:
Javascript基础教程之关键字和保留字汇总
C语言求幂计算的高效解法
使用CSS3的::selection改变选中文本颜色的方法
win10系统怎么打开摄像头或者相机?
WinXP玩大型游戏弹出虚拟内存不足如何手动调节
MySQL查询本周、上周、本月、上个月份数据的sql代码
SQL Server 数据库分离与附加(图文教程)
C 语言程序结构示例解析
Windows 10 Build 9926新图标一览:系统图标一直在变
Android Studio使用教程(六):Gradle多渠道打包
win10无法连接网络怎么办?Win10提示无法连接此网络的原因与解决方案
Xbox One系统更新至Win10一周年更新版是什么感觉?
Android Webview使用小结
JavaScript变量的作用域全解析