DCL实现单例要不要加volatile修饰

单例模式的DCL(Double-checked locking 双重校验锁定模式)写法:

public class DclSingleton {

    private static volatile DclSingleton dclSingleton = null;

    public static DclSingleton getDclSingleton(){
        if(dclSingleton == null){
            synchronized (DclSingleton.class){
                if(dclSingleton == null){
                    dclSingleton = new DclSingleton();
                }
            }
        }
        return dclSingleton;
    }
}

可以看到加了两个dclSingleton == null的判断,主要是为了提高效率,如果最外面的dclSingleton == null不要的话,每次获取对象的时候都需要进行加锁操作,比较耗时间,因为只需要第一个获取锁的创建完对象,后面就不需要在进加锁的方法里了,大大提高效率。

但是有个问题,我们的声明dclSingleton要不要加volatile修饰呢?答案是肯定需要要明白为什么,首选需要明白对象的创建过程。

public class ObjTest {
    public static void main(String[] args) {
        Object object = new Object();
    }
}

运行这个类,然后使用javap -c  ObjTest.class 命令查看生成的字节码指令。

可以看到对象的创建并非原子操作,new 出一个对象,简要的概括可以分为三步:
1.在堆上分配一块内存区域;
2.对象数据初始化;
3.将符号引用,指向堆内的实际内存地址 

而指令4和7会发生指令重排,在多线程中如果不加volatile修饰,如果发生指令重排,可以能会使用到半初始化对象,也就是说假如我第一个线程刚把对象创建完,里面的变量值还没初始化,只是赋了默认值,这时第二个线程在判断dclSingleton不等于空时,获取到的变量值可能是默认值,而不是我要设置的初始值。所以要加volatile修饰,禁止指令重排。需要等对象实例化完毕,建立了完整关系。

扫描二维码关注公众号,回复: 13139692 查看本文章


 

猜你喜欢

转载自blog.csdn.net/dhj199181/article/details/108859288