双重检验锁⽅式实现单例模式(线程安全)

public class Singleton {

private volatile static Singleton uniqueInstance;

private Singleton() {
 }

public static Singleton getUniqueInstance() {
    //先判断对象是否已经实例过,没有实例化过才进⼊加锁代码
    if (uniqueInstance == null) {
        //类对象加锁
        synchronized (Singleton.class) {
            if (uniqueInstance == null) {
                uniqueInstance = new Singleton();
                 }
             }
         }
        return uniqueInstance;
   }
}

uniqueInstance 采⽤ volatile 关键字修饰也是很有必要的, uniqueInstance = new Singleton(); 这段代 码其实是分为三步执⾏:

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

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

volatile 关键字能保证变量的可⻅性,但不能保证对变量的操作是原⼦性的,还有就是防⽌ JVM 的指 令重排序。 如果我们将变量声明为 volatile ,在对这个变量进⾏读写操作的时候,会通过插⼊特定 的 内存屏障 的⽅式来禁⽌指令重排序。

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值