面试题:双重检验锁⽅式实现 单例模式

面试题:双重检验锁⽅式实现 单例模式,第1张

面试题:双重检验锁⽅式实现 单例模式 面试题:双重检验锁方式实现 单例模式
关键词
  • 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(); 这段代码其实是分为三步执⾏:

  1. 为 instance分配内存空间
  2. 初始化 instance
  3. 将 instance指向分配的内存地址

由于 JVM 具有指令重排的特性,执⾏顺序有可能变成 1->3->2。指令重排在单线程环境下不会出现问题,但是在多线程环境下会导致⼀个线程获得还没有初始化的实例。例如,线程 T1 执⾏了 1 和 3,此时 T2 调⽤ getInstance () 后发现 instance不为空,因此返回instance,但此时 instance还未被初始化。

因此:需要使⽤ volatile 可以禁⽌ JVM 的指令重排,保证在多线程环境下也能正常运⾏

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/zaji/4660905.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-11-06
下一篇 2022-11-06

发表评论

登录后才能评论

评论列表(0条)

保存