单例模式的双重检查

1.双重检查锁定的由来

下面是非线程安全的延迟初始化对象的示例代码。

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引用的对象还没有完成初始化(原因之后分析)
对于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()被多个线程调用,将导致程序性能下降。反之,那么这个延迟初始化方案能提供令人满意的性能。

在早期的JVM中,synchronized(甚至是无竞争的synchronized)存在着巨大性能开销。因此,出现双重检查锁定。以下是示例代码。

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带来的性能开销。

这样似乎很完美,但这是一个错误的优化!在线程执行到第4行,代码读取到instance不为null时,instance引用的对象可能还没有完成初始化。

2.问题的根源

前面的双重检查示例代码第7行创建了一个对象。这一行代码可以分解为如下的3行伪代码。

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

上面3行伪代码中的2和3之间,可能会被重排序。2和3重排序之后的执行时序如下。

memory=allocate();        //1:分配对象的内存空间
instance = memory;          //3:设置instance指向刚分配的内存地址
                            //注意,此时对象还没有被初始化!
ctorInstance(memory);     //2:初始化对象
单例模式的双重检查_第1张图片
多线程执行时序图

由于单线程内要遵守intra-thread semantics,从而能保证A线程的执行结果不会被改变。但是,当线程A和B按上图时序执行时,B线程将看到一个还没有被初始化的对象。

回到主题,DoubleCheckedLocking代码第7行(instance=new Instance();)如果发生重排序,拎一个并发执行的线程B就有可能在第4行判断instance不为null。线程B接下来访问instance所引用的对象,但此时这个对象可能还没有被A线程初始化!

在知晓了问题发生的根源之后,我们可以想出两个办法来实现线程安全的延迟初始化。
1.不允许2和3重排序
2.允许2和3重排序,但不允许其他线程“看到”这个重排序。
基于上面这两点,提出两个解决方案。

3.1基于volatile的解决方案

对于前面的基于双重检查锁定来实现延迟初始化的方案,只需要做一点小的修改(把instance声明为volatile型),就可以实现线程安全的延迟初始化。请看下面的示例代码。

public class SafeDoubleCheckedLocking {
    private volatile static Instance instance;
    public  static Instance getInstance(){
        if(instance ==null) {
            synchronized (SafeDoubleCheckedLocking.class) {
                if (instance == null)
                    instance = new Instance();              //instance为volatile,现在没问题了
            }
        }
        return instance;
    }
}

当声明对象的引用为volatile后,之前的3行伪代码中的2和3之间的重排序,在多线程环境中将会被禁止。上面的示例代码江安如下的时序执行。

单例模式的双重检查_第2张图片
多线程执行时序图

这个方案是通过禁止上图2和3之间的重排序,来保证线程安全的延迟初始化。

3.2基于类初始化的解决方案

JVM在类的初始化阶段(即在Class被加载后,且被线程使用之前),会执行类的初始化。在执行类的初始化期间,JVM会去获取一个锁。这个锁可以同步多个线程对同一个类的初始化。

基于这个特性可以实现另一种线程安全的延迟初始化方案。

public class InstanceFactory{
    private static class InstanceHolder{
        public static Instance instance = new Instance();
    }

    public static Instance getInstance(){
        return InstanceHolder.instance;         //这里将导致InstanceHolder类被初始化
    }
}

假设两个线程并发执行getInstance()方法,下面是执行示意图。

单例模式的双重检查_第3张图片
两个线程并发执行的示意图

这个方案的实质是:允许之前的3行伪代码中的2和3重排序,但不允许非构造线程(这里指线程B)“看到”这个重排序。

4.总结

通过对比基于volatile的双重检查锁定的方案和基于类初始化的方案,我们会发现基于类初始化的方案的实现代码更简洁。但基于volatile的双重检查锁定的方案有一个额外的优势:除了可以对静态字段实现延迟初始化外,还可以对实例字段实现延迟初始化。

字段延迟初始化降低了初始化类或创建实例的开销,但增加了访问被延迟初始化的字段的开销。在大多数时候,正常的初始化要优于延迟初始化。如果确实需要对实例字段使用线程安全的延迟初始化,请使用上面介绍的基于volatile的延迟初始化方案;如果确实需要对静态字段使用线程安全的延迟初始化,请使用基于类初始化的方案。

——摘自《Java并发编程的艺术

你可能感兴趣的:(单例模式的双重检查)