双重校验锁实现对象单例

该博客详细介绍了如何在Java中使用双重校验锁(Double-Check Locking)实现线程安全的单例模式。通过volatile关键字确保内存可见性和防止指令重排序,以避免多线程环境下可能出现的问题,如未初始化的实例被返回。这种方法既保证了效率,也确保了单例的正确创建。
摘要由CSDN通过智能技术生成

双重校验锁实现对象单例

package com.heu.wsq.basic;

/**
 * 双重校验所锁实现对象单例(线程安全)
 * @author wsq
 * @date 2021/1/24
 */
public class Singleton {
    /**
     * volatile作用:
     * 1.保证内存可见性
     * 2.防止指令重排序
     */
    private volatile static Singleton uniqueInstance;

    private Singleton(){}

    public static Singleton getInstance(){
        // 判断对象是否已经实例过,没有实例化过才进入加锁代码
        if (uniqueInstance == null){
            // 由于singleton属于这个类,因此给Singleton加锁
            synchronized(Singleton.class){
                if (uniqueInstance == null){
                    uniqueInstance = new Singleton();
                }
            }
        }
        return uniqueInstance;
    }
}

为什么要使用volatile修饰uniqueInstance?

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

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

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

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值