双重检查锁定

看 "java并发编程的艺术" 第3.8

双重检查锁定与延迟初始化

  在Java多线程程序中,有时候需要采用延迟初始化来降低初始化类和创建对象的开销。双
重检查锁定是常见的延迟初始化技术,但它是一个错误的用法。本文将分析双重检查锁定的
错误根源,以及两种线程安全的延迟初始化方案。

需要注意的是, 双重检查锁定本身是错误的!

双重检查锁定的由来

  在Java程序中,有时候可能需要推迟一些高开销的对象初始化操作,并且只有在使用这些
对象时才进行初始化。此时,程序员可能会采用延迟初始化。但要正确实现线程安全的延迟初
始化需要一些技巧,否则很容易出现问题。比如,下面是非线程安全的延迟初始化对象的示例
代码。

public class UnsafeLazyInitialization {
    private static Instance instance;

    public static Instance getInstance() {
        if (instance == null) { // 1:A线程执行
            instance = new Instance(); // 2:B线程执行
        }

        return instance;
    }
}

  

  在UnsafeLazyInitialization类中,假设A线程执行代码1的同时,B线程执行代码2。此时,线
程A可能会看到instance引用的对象还没有完成初始化(出现这种情况的原因见3.8.2节)。
  对于UnsafeLazyInitialization类,我们可以对getInstance()方法做同步处理来实现线程安全
的延迟初始化。示例代码如下。

public class SafeLazyInitialization {
    private static Instance instance;

    public synchronized static Instance getInstance() {
        if (instance == null) {
            instance = new Instance();
        }

        return instance;
    }
}

  

  由于对getInstance()方法做了同步处理,synchronized将导致性能开销。如果getInstance()方
法被多个线程频繁的调用,将会导致程序执行性能的下降。反之,如果getInstance()方法不会被
多个线程频繁的调用,那么这个延迟初始化方案将能提供令人满意的性能。
  在早期的JVM中,synchronized(甚至是无竞争的synchronized)存在巨大的性能开销。因此,
人们想出了一个“聪明”的技巧:双重检查锁定(Double-Checked Locking)。人们想通过双重检查
锁定来降低同步的开销。下面是使用双重检查锁定来实现延迟初始化的示例代码。

public class DoubleCheckedLocking { // 1

    private static Instance instance; // 2

    public static Instance getInstance() { // 3

        if (instance == null) { // 4:第一次检查

            synchronized (DoubleCheckedLocking.class) { // 5:加锁

                if (instance == null) { // 6:第二次检查
                    instance = new Instance(); // 7:问题的根源出在这里
                }
            } // 8
        } // 9

        return instance; // 10
    } // 11
}

  

  如上面代码所示,如果第一次检查instance不为null,那么就不需要执行下面的加锁和初始
化操作。因此,可以大幅降低synchronized带来的性能开销。上面代码表面上看起来,似乎两全
其美。
  多个线程试图在同一时间创建对象时,会通过加锁来保证只有一个线程能创建对象。
·在对象创建好之后,执行getInstance()方法将不需要获取锁,直接返回已创建好的对象。
双重检查锁定看起来似乎很完美,但这是一个错误的优化!在线程执行到第4行,代码读
取到instance不为null时,instance引用的对象有可能还没有完成初始化。

3.8.2 问题的根源

前面的双重检查锁定示例代码的第7行(instance=new Singleton();)创建了一个对象。这一
行代码可以分解为如下的3行伪代码。

memory = allocate();  // 1:分配对象的内存空间
ctorInstance(memory); // 2:初始化对象
instance = memory;  // 3:设置instance指向刚分配的内存地址

上面3行伪代码中的2和3之间,可能会被重排序(在一些JIT编译器上,这种重排序是真实
发生的,详情见参考文献1的“Out-of-order writes”部分)。2和3之间重排序之后的执行时序如
下。

memory = allocate();  // 1:分配对象的内存空间
instance = memory;  // 3:设置instance指向刚分配的内存地址
// 注意,此时对象还没有被初始化!
ctorInstance(memory); // 2:初始化对象

  

  上面3行伪代码的2和3之间虽然被重排序了,但这个重排序
并不会违反intra-thread semantics。这个重排序在没有改变单线程程序执行结果的前提下,可以
提高程序的执行性能。
为了更好地理解intra-thread semantics,请看如图3-37所示的示意图(假设一个线程A在构
造对象后,立即访问这个对象)。
如图3-37所示,只要保证2排在4的前面,即使2和3之间重排序了,也不会违反intra-thread
semantics。
下面,再让我们查看多线程并发执行的情况。如图

原文地址:https://www.cnblogs.com/jiuya/p/10792466.html

时间: 2024-08-03 17:23:40

双重检查锁定的相关文章

双重检查锁定与延迟初始化

在Java程序中,有时候可能需要推迟一些高开销的对象初始化操作,并且只有在使用这些对象时才进行初始化.此时程序员可能会采用延迟初始化.但要正确实现线程安全的延迟初始化需要一些技巧,否则很容易出现问题.比如,下面是非线程安全的延迟初始化对象的示例代码: public class UnsafeLazyInitialization { private static Instance instance; public static Instance getInstance() { if (instanc

JAVA 双重检查锁定和延迟初始化

双重检查锁定的由来在Java程序中,有时需要推迟一些高开销的对象的初始化操作,并且只有在真正使用到这个对象的时候,才进行初始化,此时,就需要延迟初始化技术.延迟初始化的正确实现是需要一些技巧的,否则容易出现问题,下面一一介绍. 方案1 public class UnsafeLazyInit{ private static Instance instance; public static Instance getInstance(){ if (instance == null){ instance

双重检查锁定的单例模式和延迟初始化

有时候需要推迟一些高开销的对象初始化操作,并且只有在使用这些对象时才进行初始化.此时,常用的可能就是延迟初始化,例如:懒汉式单例模式,但是要正确的实现线程安全的延迟初始化需要一些技巧,下面是非线程安全的示例代码: public class UnsafeLazyInit { private static Instance instance ; public static Instance getInstance(){ if(instance == null ) //1.A线程执行 instance

双重检查锁定的由来

双重检查锁定的由来 在java程序中,有时候可能需要推迟一些高开销的对象初始化操作,并且只有在使用这些对象时才进行初始化.此时程序员可能会采用延迟初始化.但要正确实现线程安全的延迟初始化需要一些技巧,否则很容易出现问题.比如,下面是非线程安全的延迟初始化对象的示例代码: public class UnsafeLazyInitialization { private static Instance instance; public static Instance getInstance() { i

双重检查锁定原理详解

双重检查锁定与延迟初始化 在java程序中,有时候可能需要推迟一些高开销的对象初始化操作,并且只有在使用这些对象时才进行初始化.此时程序员可能会采用延迟初始化.但要正确实现线程安全的延迟初始化需要一些技巧,否则很容易出现问题.比如,下面是非线程安全的延迟初始化对象的示例代码: public class UnsafeLazyInitialization { private static Instance instance; public static Instance getInstance()

Singleton(单例)模式和Double-Checked Locking(双重检查锁定)模式

问题描述 现在,不管开发一个多大的系统(至少我现在的部门是这样的),都会带一个日志功能:在实际开发过程中,会专门有一个日志模块,负责写日志,由于在系统的任何地方,我们都有可能要调用日志模块中的函数,进行写日志.那么,如何构造一个日志模块的实例呢?难道,每次new一个日志模块实例,写完日志,再delete,不要告诉我你是这么干的.在C++中,可以构造一个日志模块的全局变量,那么在任何地方就都可以用了,是的,不错.但是,我所在的开发部门的C++编码规范是参照Google的编码规范的. 全局变量在项目

Java单例模式中双重检查锁的问题

单例创建模式是一个通用的编程习语.和多线程一起使用时,必需使用某种类型的同步.在努力创建更有效的代码时,Java 程序员们创建了双重检查锁定习语,将其和单例创建模式一起使用,从而限制同步代码量.然而,由于一些不太常见的 Java 内存模型细节的原因,并不能保证这个双重检查锁定习语有效. 它偶尔会失败,而不是总失败.此外,它失败的原因并不明显,还包含 Java 内存模型的一些隐秘细节.这些事实将导致代码失败,原因是双重检查锁定难于跟踪.在本文余下的部分里,我们将详细介绍双重检查锁定习语,从而理解它

为什么双重检查锁模式需要 volatile ?

双重检查锁定(Double check locked)模式经常会出现在一些框架源码中,目的是为了延迟初始化变量.这个模式还可以用来创建单例.下面来看一个 Spring 中双重检查锁定的例子. 这个例子中需要将配置文件加载到 handlerMappings中,由于读取资源比较耗时,所以将动作放到真正需要 handlerMappings 的时候.我们可以看到 handlerMappings 前面使用了volatile .有没有想过为什么一定需要 volatile?虽然之前了解了双重检查锁定模式的原理

双重检查锁单例模式为什么要用volatile关键字?

前言 从Java内存模型出发,结合并发编程中的原子性.可见性.有序性三个角度分析volatile所起的作用,并从汇编角度大致说了volatile的原理,说明了该关键字的应用场景:在这补充一点,分析下volatile是怎么在单例模式中避免双检锁出现的问题的. 并发编程的3个条件 1.原子性:要实现原子性方式较多,可用synchronized.lock加锁,AtomicInteger等,但volatile关键字是无法保证原子性的:2.可见性:要实现可见性,也可用synchronized.lock,v