单例模式之DCL懒汉式解析(双重检验锁)

单例模式之DCL懒汉式解析(双重检验锁)

双重检验锁不安全的原因

  • 在下面的代码进行到: lazyMan = new LazyMan();的时候有可能会进行指令重排.
  • lazyMan = new LazyMan(); 这一句代码其实有3个操作
    1. 分配内存空间
    2. 执行构造方法 初始化对象
    3. 把对象指向 内存空间
  • A 线程的执行顺序可能是1 3 2, 并且由于不是原子性操作过程, A在操作的过程会可能会有其他线程来执行操作
    假如A 执行完3 还没执行2的时候, 线程B进来执行在第一个if (lazyMan == null)的地方, 由于A执行完3, 会认为lazyMan不为null(因为已经指向了内存空间) , 所以B会返回未完成构造的lazyMan会有问题, 需要在lazyMan上加volatile修饰
// 懒汉式: 用的时候创建
public class LazyMan {
    private LazyMan() {
    }
	
	// volatile保证lazyMan创建过程中不进行指令重排
    private volatile static LazyMan lazyMan;



    // 双重检验锁 DCL懒汉
    public static LazyMan getLazyMan() {
        if (lazyMan == null) {
        	// 获取LazyMan这个类的锁
            synchronized (LazyMan.class) {
                if (lazyMan == null) {
                    try {
                        TimeUnit.SECONDS.sleep(1);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    lazyMan = new LazyMan();
                    /**
                     * 不是原子性操作
                     * 1. 分配内存空间
                     * 2. 执行构造方法 初始化对象
                     * 3. 把对象指向 内存空间
                     *
                     * 会进行指令重排
                     * A 线程的执行顺序可能是1 3 2
                     *   当A 执行完3 还没执行2的时候  线程B进来执行  由于执行完3 会认为lazyMan
                     *   不为null(因为已经指向了内存空间) 所以B会返回未完成构造的lazyMan
                     *   需要在lazyMan上加volatile修饰
                     */

                }
            }
        }
        return lazyMan;
    }


    // 模拟多线程
    public static void main(String[] args) {
        for (int i = 0; i < 10; i++) {
            new Thread(() -> {
                System.out.println(Thread.currentThread().getName() + "lazyMan : " + getLazyMan());
            }, "" + i).start();
        }
    }
}
  • 控制台输出: 同一个实例
0lazyMan : com.zheng.juc.single.LazyMan@42b95647
9lazyMan : com.zheng.juc.single.LazyMan@42b95647
4lazyMan : com.zheng.juc.single.LazyMan@42b95647
6lazyMan : com.zheng.juc.single.LazyMan@42b95647
5lazyMan : com.zheng.juc.single.LazyMan@42b95647
2lazyMan : com.zheng.juc.single.LazyMan@42b95647
1lazyMan : com.zheng.juc.single.LazyMan@42b95647
7lazyMan : com.zheng.juc.single.LazyMan@42b95647
3lazyMan : com.zheng.juc.single.LazyMan@42b95647
8lazyMan : com.zheng.juc.single.LazyMan@42b95647
  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值