关键词
- 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 的指令重排,保证在多线程环境下也能正常运⾏
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)