面试题:双重检验锁方式实现 单例模式
关键词
- volatile 禁⽌ JVM 中 构造方法的 指令重排
编码实现
public class Singleton {
private volatile static Singleton instance;
private Singleton() {
}
public static Singleton getInstance() {
//先判断对象是否已经实例过,没有实例化过才进⼊加锁代码
if (instance== null) {
//类对象加锁
synchronized (Singleton.class) {
if (instance== null) {
instance= new Singleton(); //构造方法指令重排问题
}
}
}
return instance;
}
}
注意:instance 采用 volatile 关键字修饰是很有必要
instance= new Singleton(); 这段代码其实是分为三步执⾏:
- 为 instance分配内存空间
- 初始化 instance
- 将 instance指向分配的内存地址
由于 JVM 具有指令重排的特性,执⾏顺序有可能变成 1->3->2。指令重排在单线程环境下不会出现问题,但是在多线程环境下会导致⼀个线程获得还没有初始化的实例。例如,线程 T1 执⾏了 1 和 3,此时 T2 调⽤ getInstance () 后发现 instance不为空,因此返回instance,但此时 instance还未被初始化。
因此:需要使⽤ volatile 可以禁⽌ JVM 的指令重排,保证在多线程环境下也能正常运⾏