• Double Check Lock


    Double Check Lock(DCL)


    通过单例模式生产类是程序员必会,它有很多写法,其中的懒汉式,及延迟生成类,应使用双重检查,否则就会出现生成多例:

    public class Singleton {
        private static Singleton singleton;
        private Singleton() {}
        public Singleton getInstance() {
            if (singleton == null) {
                synchronized (Singleton.class){
                    if(singleton == null) {
                        singleton = new Singleton();
                    }
                }
            }
            return singleton;
        }
    }
    

    以上代码看起来似乎以及完美了,但是其实还有漏洞。如下:

    实例化一个对象要分为三个步骤:

    1. 分配内存空间
    2. 初始化对象
    3. 将内存空间的地址赋值给对应的引用

    但是由于重排序的缘故,步骤2、3可能会发生重排序,其过程如下:

    1. 分配内存空间
    2. 将内存空间的地址赋值给对应的引用
    3. 初始化对象

    如果2、3发生了重排序就会导致第二个判断会出错,singleton != null,但是它其实仅仅只是一个地址而已,此时对象还没有被初始化,所以return的singleton对象是一个没有被初始化的对象。

    解决方法是在一处加上关键字:

    private volatile static Singleton singleton;       //1
    

    这样就保证了变量singleton的可见性,使得程序能到达到我们的预期。


    参考资料:

    http://cmsblogs.com/?p=2161

  • 相关阅读:
    生产者消费者问题--进阶
    互斥量和信号量的区别
    linux多线程大神博客网址
    生产者消费者
    文件互斥
    Linux中link,unlink,close,fclose详解
    条件变量
    哲学家进餐问题-3中解决方案
    使用读写锁解决读者-写者问题
    架构漫谈(三):如何做好架构之识别问题
  • 原文地址:https://www.cnblogs.com/fruitknife/p/9703083.html
Copyright © 2020-2023  润新知