双重检查锁定实现的单例模式,有什么问题?如何修复?

目录

面试题:

大多数面试者的回答(错误或不完整):

正确答案要点(面试官真正想听的):

问题本质:指令重排序导致"部分初始化"对象被泄露

后果:其他线程可能拿到一个"未完全初始化"的对象

[修复方案:用 volatile 禁止重排序](#修复方案:用 volatile 禁止重排序)

Bonus:更优解(推荐)

面试官考察点:

如果你在面试中这样回答:


面试题:

下面这段双重检查锁定(Double-Checked Locking)实现的单例模式,有什么问题?如何修复?

复制代码
public class Singleton {
    private static Singleton instance;

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

大多数面试者的回答(错误或不完整):

"没问题啊,加了 synchronized,是线程安全的。"

或者:

"应该加 volatile,防止指令重排序。"

------但为什么需要 volatile?不加到底会发生什么?很多人说不清楚。


问题本质:指令重排序导致"部分初始化"对象被泄露

instance = new Singleton(); 这行代码在 JVM 中并非原子操作,实际分为三步:

  1. 分配内存空间;
  2. 调用构造函数,初始化对象;
  3. instance 引用指向分配的内存地址。

但步骤 2 和 3 可能被 JIT 编译器重排序为:1 → 3 → 2!

后果:其他线程可能拿到一个"未完全初始化"的对象
  • 线程 A 执行到步骤 3(instance 已非 null),但尚未完成构造(步骤 2);
  • 线程 B 进入 getInstance(),发现 instance != null,直接返回;
  • 线程 B 使用这个对象时,可能访问到未初始化的字段(默认值),导致 NPE 或逻辑错误!

这不是理论风险!在高并发 + 特定 CPU 架构(如 ARM)下极易复现。

修复方案:用 volatile 禁止重排序
复制代码
private static volatile Singleton instance; // 关键!
  • volatilehappens-before 语义保证:
    • 写操作(步骤 3)前的所有操作(包括构造函数)对读线程可见;
    • 禁止步骤 2 和 3 的重排序。
Bonus:更优解(推荐)
  • 静态内部类(利用类加载机制)

    复制代码
    public class Singleton {
        private static class Holder {
            static final Singleton INSTANCE = new Singleton();
        }
        public static Singleton getInstance() {
            return Holder.INSTANCE;
        }
    }
    • 线程安全、懒加载、无锁、性能高,且无需处理重排序问题。
相关推荐
重生之我是Java开发战士5 天前
【Java SE】多线程(三):单例模式,阻塞队列,线程池与定时器
java·javascript·单例模式
许彰午6 天前
34_Java设计模式之单例模式
java·单例模式·设计模式
罗超驿8 天前
10.Java单例模式全解析:饿汉式与懒汉式实现及线程安全深度剖析
安全·单例模式·javaee
布朗克1688 天前
33 设计模式精讲
java·单例模式·设计模式
雨浓YN8 天前
基于设计模式的Winform软件框架-01Xml\Log\Ini日志(单例模式+生产者消费者模式)
单例模式·设计模式
仙俊红9 天前
Java 单例模式:类里面为什么可以有自己类型的字段?
java·开发语言·单例模式
swordbob9 天前
prototype 注入到 singleton 里,prototype是否还是线程安全的
安全·spring·单例模式·原型模式
谁似人间西林客11 天前
工业大数据实战:看中国智造如何用数据驱动效率革命
大数据·单例模式
张小姐的猫11 天前
【Linux】多线程 —— 线程池 | 单例模式 | 常见锁
linux·运维·服务器·c++·单例模式·设计模式·策略模式
Java面试题总结12 天前
双重检验锁的单例模式在高并发下的可见性问题
单例模式