`
ITChaser
  • 浏览: 7806 次
  • 性别: Icon_minigender_1
  • 来自: 太原
社区版块
存档分类
最新评论

"double-checked locking"是否真的安全呢

阅读更多
本文的思想来源于:
Jeremy Manson and Brian Goetz, 《JSR 133 (Java Memory Model) FAQ》, February 2004, http://www.cs.umd.edu/~pugh/java/memoryModel/jsr-133-faq.html#synchronization

"double-checked locking"经常被用于Singleton模型中实现lazy load,典型的"double-checked locking"代码如下所示:

  // double-checked-locking - don't do this!
		
		// initialization
		private static Something instance = null;
		
		public static Something getInstance() {
		  if (instance == null) {
		    synchronized (this) {
		      if (instance == null)
		        instance = new Something();
		    }
		  }
		  return instance;
		}


然后由于reordering的存在,instance 的初始化有可能会被move到getInstance之后执行。这种情况下,
虽然在getInstance()方法中,instance被正确的赋值,但是之后又被初始化为null。下面的程序打印的结果为null。

  public static Something getInstance() {
		  if (instance == null) {
		    synchronized (this) {
		      if (instance == null)
		        instance = new Something();
		    }
		  }
		  return instance;
		}
		
		private static Something instance = null;
		
		public static void main(String[] args) {
			Something instance = getInstance();
			
			System.out.println(instance);
		}


由于volatile的特性(参加另一篇文章《volatile关键词的原理与用途》),这个问题显然可以通过将instance字段声明为volatile来fix。但是更为简单直观的方法为:

  public Class LazySomething{
			// static inner class
			private static class LazySomethingHolder {
			  public static Something something = new Something();
			}
			
			public static Something getInstance() {
			  return LazySomethingHolder.something;
			}
		}
0
0
分享到:
评论

相关推荐

    C++ and the Perils of Double-Checked Locking

    在介绍双检锁模式(Double-Checked Locking Pattern,DCLP)的C++实现中,Scott Meyers和Andrei Alexandrescu在其2004年的文章中指出,传统的单例模式实现并不具备线程安全性。单例模式是设计模式中经常被提及的一种...

    C++ and the Perils of Double Checked Locking.zip

    《C++ and the Perils of Double Checked Locking》是一篇探讨C++编程中双重检查锁定(Double-Checked Locking)模式潜在问题的文献。在多线程编程中,双重检查锁定是一种常见的优化策略,旨在减少对同步原语的依赖...

    c++ and Peris of Double Checked Locking

    为了解决单例模式的线程安全问题,一种流行的方法是采用双检查锁定模式(Double-Checked Locking Pattern,简称DCLP)。DCLP的目标是在初始化共享资源(如单例对象)时添加高效的线程安全性,通过在检查实例是否存在...

    深入剖析Java中的双检锁模式:实现、陷阱与最佳实践

    在Java并发编程中,双检锁(Double-Checked Locking)是一种用于减少同步开销的优化技术,尤其适用于懒加载(lazy initialization)的场景。本文将详细探讨双检锁的工作原理、潜在问题以及如何安全地实现它。 双检锁...

    ACE中的DoubleCheckedLocking模式

    北京火龙果软件工程技术中心意图无论什么时候当临界区中的代码仅仅需要加锁一次,同时当其获取锁的时候必须是线程安全的,可以用DoubleCheckedLocking模式来减少竞争和加锁载荷。动机1、标准的单例。开发正确的有效...

    Java多线程之延迟初始化1

    synchronized (DoubleCheckedLocking.class) { if (instance == null) { instance = new Instance(); } } } return instance; } } ``` 双重检查锁定的关键在于使用`volatile`关键字修饰`instance`变量。`...

    C++CLI中实现singleton模式

    双重检测锁(Double-Checked Locking)实现的Singleton模式在多线程应用中有相当的价值。在ACE的实现中就大量使用ACE_Singleton模板类将普通类转换成具有Singleton行为的类。这种方式很好地消除了一些重复代码臭味,...

    设计模式解析-英文

    各种工厂模式 242 第21章 Singleton模式和Double-Checked Locking模式 249 第22章 Object Pool模式 257 第23章 Factory Method模式 267 第24章 工厂模式的总结 272 第八部分 终点与起点 第25章 设计模式回顾:总结与...

    单例模式的5种设计.docx

    3. **双检锁/双重校验锁(DCL,Double-Checked Locking)**: 双检锁模式结合了懒汉式的延迟加载和饿汉式的线程安全性。在多线程环境中,首次检查实例是否存在,若不存在再进行同步操作,创建实例。这种模式减少了...

    Java多线程编程环境中单例模式的实现

    为了提高性能,人们提出了**双重检查锁定**(Double-checked locking)的方法。这种方法首先在不加锁的情况下检查`instance`是否为`null`,如果为`null`则进行同步操作: ```java public static Singleton ...

    Java并行(4):线程安全前传之Singleton1

    3. 双检锁(Double-Checked Locking) 为提高性能,可以采用双检锁(DCL)策略。在DCL模式中,首先检查`instance`是否为null,如果不是则直接返回,只有当`instance`为null时,才进入同步块再次检查。这样大部分情况...

    java之 ------ 几种常见的简单设计模式

    - 可以通过双重检查锁定(Double-Checked Locking)优化性能。 **线程安全问题及解决方案** - **问题**:在多线程环境中,若不采取措施,则可能会导致单例模式失效,即生成多个实例。 - **解决方案**: - 使用`...

    java面试题_源码解读(3题)

    - **双重检查锁定(Double-Checked Locking)** 是一种优化的单例模式实现,它在多线程环境下确保了线程安全的同时减少了同步开销。 - 关键在于 volatile 关键字,它保证了多个线程对实例的可见性和初始化的正确...

    Java面试.docx

    3. **双端检锁(Double-Checked Locking, DCL)机制**: - DCL是一种尝试提高单例模式效率的设计模式,但如果没有正确同步,可能存在指令重排问题,导致非线程安全。 - 使用volatile可以禁止指令重排,从而在一定...

    Java双重检查加锁单例模式的详解

    DCL(Double-checked locking)是Java双重检查加锁单例模式的一种实现方法。它使用了synchronized关键字来确保线程安全,但是这也会带来性能损失。DCL看起来是一个聪明的优化,但是它却不能保证正常工作。 在多线程...

    Java-设计模式-单例模式-实现源码(简单实现、双重检查锁、静态内部类、枚举类)

    在Java中,有多种实现单例模式的方法,包括简单实现、双重检查锁定(Double-Checked Locking)、静态内部类和枚举类。下面我们将详细探讨这些不同的实现方式。 1. **简单实现(非线程安全)** 最简单的单例实现...

    单例模式简介和java代码实现

    }}在双重检查锁(Double-Checked Locking)实现中,第一次检查是在不加锁的情况下进行的,只有当第一次检查后 instance 仍然为 null 时,才会进入同步代码块进行第二次检查和实例化。这种方式提高了并发性能,因为...

    一例读懂设计模式-单例模式、简单工厂模式.zip

    - 双重检查锁定(Double-Checked Locking):在多线程环境下,确保单例在被多次请求时仍保持唯一性,同时减少同步开销。 - 饿汉式(静态常量):在类加载时立即创建实例,保证线程安全,但可能会浪费内存资源。 -...

    Singleton模式源程序

    2. 双重检查锁定(Double-Checked Locking):这是最常见的一种线程安全实现方式。在获取实例时先检查实例是否已经存在,若不存在再加锁并创建。这样可以减少不必要的同步开销。 3. 原子操作:利用C++11引入的std::...

    Java并行(3):可见性重访之锁、Volatile与原子变量1

    尽管如此,`volatile`在某些场景下,如单例模式的双重检查锁定(Double-Checked Locking)或者作为标志位时,可以有效提升性能,同时保证基本的可见性。 总结来说,Java中的锁和`volatile`关键字都是为了应对并发...

Global site tag (gtag.js) - Google Analytics