双检锁单例的分析

首先要关注并发编程中的三大特性

  1. 线程切换带来的原子性问题
  2. 指令重排序优化 (填充流水线, 尽最大可能发挥指令级并行的特性), 以及内存写回延迟 导致的顺序性问题.
  3. 多核多cache导致的可见性问题

针对上面三种问题来分析双检锁的实现方式

public class Singleton {

    private static volatile Singleton instance;
    
    private static Singleton getInstance() {
        if (instance == null) {
            synchronized (Singleton.class) {
                if (instance == null) {
                    instance = new Singleton();
                }
            }
        }

        return instance;
    }
}

首先, 原子性问题的实现是由synchronized关键字来保证, synchronized 保证对象初始化操作是线程间互斥的, 不会在一个线程new Singleton的同时, 另一个线程也进入这个过程中, 从而产生多个单例

可见性问题也是由 synchronized 来保证(volatile来保证也有道理…), synchronized 实现了两条语义
1. 线程加锁时, 将工作内存置无效, 共享变量都从工作内存读取
2. 线程解锁时, 将共享变量刷新到主内存

有序性 问题由 volatile 保证, 禁止指令重排序, 将instance的赋值操作固定到init操作之后

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值