Java单例模式双重检查锁定中volatile关键字的作用

首先要说的是,个人推荐使用饿汉模式和静态内部类方式实现单例模式其实,静态内部类里也是一个饿汉模式。
懒汉模式中,双重检查锁定代码如下:

public class Singleton{   
    // 静态属性,volatile保证可见性和禁止指令重排序
    private volatile static Singleton instance = null; 
    // 私有化构造器  
    private Singleton(){}   

    public static  Singleton getInstance(){   
        // 第一重检查锁定
        if(instance==null){  
            // 同步锁定代码块 
            synchronized(Singleton.class){
                // 第二重检查锁定
                if(instance==null){
                    // 注意:非原子操作
                    instance=new Singleton(); 
                }
            }              
        }   
        return instance;   
    }   
}

volatile作用:以下会涉及到Java内存模型的知识

  • 禁止指令重排序。我们知道new Singleton()是一个非原子操作,编译器可能会重排序【构造函数可能在整个对象初始化完成前执行完毕,即赋值操作(只是在内存中开辟一片存储区域后直接返回内存的引用)在初始化对象前完成】。而线程B在线程A赋值完时判断instance就不为null了,此时B拿到的将是一个没有初始化完成的半成品

  • 保证可见性。线程A在自己的工作线程内创建了实例,但此时还未同步到主存中;此时线程B在主存中判断instance还是null,那么线程B又将在自己的工作线程中创建一个实例,这样就创建了多个实例

顺便提一下,volatile禁止指令重排序只能保证volatile修饰的代码之后的代码不会在它之前执行。

猜你喜欢

转载自blog.csdn.net/zcl_love_wx/article/details/80758162
今日推荐