DCL单例要不要加volatile关键字?
首先得知道volatile关键字的作用,其次什么是单例,最后DCL是怎么实现的单例?
(一)
volatile关键字在java中用来修饰成员变量,有两个语义:
- 被volatile修饰成员变量,在多个线程之间可见
- volatile禁止指令重排序
(二)
单例,在程序中某个类的实例有且只能有一个。
(三)DCL,Double Check Lock的首字母缩写,实现单例的一种方式
上代码:
public class B {
private volatile static B b = null; // 这里的volatile要不要加???
private int a = 10;
private B() {
}
public static B getInstance() {
if (b == null) {
synchronized (B.class) {
if (b == null) {
b = new B();
}
}
}
return b;
}
}
场景:现在有两个线程t1和t2都调用了getInstance()
当t1和t2都执行完if(b == null) 后t1失去了cpu的执行权,t2获得锁执行了b = new B() ,上篇文章我们聊到了JVM遇到new关键字会在JVM字节码层面会执行三个指令①new ②invokespecial ③atore_n ,cpu在执行指令的时候可能会发生重排序,导致③先于②执行,以至于这个时候的b已经指向了堆内存中的对象,但是b的成员变量都是默认值,之后t2线程释放锁。就在这个时候t1线程获得了cpu的执行权,发现b已经不是null了,直接return b 那这个时候t1线程所获取到的b对象其实是半初始化的,因此必须加volatile关键字来禁止在创建B对象的时候发生指令重排序!