Java中的双重检查(Double-Check)详解

作者:88250 时间:2023-02-23 21:05:23 

在 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,双重检查
0
投稿

猜你喜欢

  • Java如何跳过https的ssl证书验证详解

    2023-08-24 11:34:56
  • flutter日期选择器 flutter时间选择器

    2023-09-22 04:50:15
  • Android下拉刷新控件PullToRefresh实例解析

    2022-01-27 08:01:14
  • Android转场效果实现示例浅析

    2023-09-21 12:10:17
  • Dubbo扩展点SPI实践示例解析

    2021-12-14 12:56:45
  • Java实现配置加载机制

    2023-11-26 09:03:38
  • Flutter 图片开发核心技能快速掌握教程

    2023-08-16 22:01:51
  • android自定义view之模拟qq消息拖拽删除效果

    2023-01-29 11:48:34
  • Mybatis参数传递示例代码

    2023-05-20 09:43:08
  • SpringBoot+Hutool+thymeleaf完成导出Excel的实现方法

    2023-09-05 17:39:10
  • java导出Excel文件的步骤全纪录

    2021-10-04 11:18:38
  • Android入门之使用SharedPreference存取信息详解

    2023-09-09 10:25:53
  • C#实现将应用程序设置为开机启动的方法

    2023-06-04 05:59:26
  • 浅谈Android ASM自动埋点方案实践

    2021-10-19 22:01:18
  • C#设置MDI子窗体只能弹出一个的方法

    2022-03-19 23:40:11
  • Android UI使用HTML布局方法实例

    2023-04-07 09:14:11
  • IntelliJ IDEA的代码搁置功能实现

    2022-03-28 07:32:52
  • 分布式调度XXL-Job整合Springboot2.X实战操作过程(推荐)

    2023-11-23 09:43:38
  • SpringCloud远程服务调用三种方式及原理

    2023-10-16 07:21:19
  • Maven实战之搭建Maven私服和镜像的方法(图文)

    2023-11-27 22:27:06
  • asp之家 软件编程 m.aspxhome.com