双检查锁单例模式

单例模式,是程序员最熟悉的模式之一。今天说一下双检查锁的实现。话不多说,上代码:


public class Singleton {
    private volatile static Singleton singleton;
    private Singleton(){}

    public static Singleton getInstance() {
        if (singleton == null) {
            synchronized (Singleton.class) {
                if (singleton == null) {
                    singleton = new Singleton();
                }
            }
        }
        return singleton;
    }
}

那么此写法是如何能确保单例,实现线程安全的呢?

首先:volatile 修饰singleton 保证了多线程环境下,禁止了JVM对指令进行重排序,同时该变量的可见性。

其次,第一次检查singleton是否为空,可以保证只有第一次进行初始化时,大量线程同时竞争Singleton 类锁而进入阻塞状态,一旦singleton 初始化完成,则将会跳过竞争。但是如果去除该处检查,则每次都将进行锁的竞争。

再次,一旦有线程获取到锁,进行二次检查,避免在此线程前,当前线程属于第一次竞争锁在处于阻塞被唤醒的线程进入代码块再次初始化singleton 而导致产生多例。

以上,DCL实现的单例,volatile 禁止了指令重排序,保证了单例共享变量在多线程环境下的可见性,同时第一检查,大大减少了因为竞争锁而导致的大量现场阻塞,提高了并发能力,同时第二次检查,真正实现了单例,保证了线程安全。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值