JAVA面试-并发篇 03-使用synchronized doublecheck实现单例有什么坑

DCL(Double-Check Locking 双重检查锁)单例的坑

一、核心结论

经典DCL最致命的是volatile缺失导致的半初始化对象空指针

二、错误 vs 正确完整写法

❌ 错误写法(90%的人会写错,缺volatile)

java 复制代码
public class Singleton {
    // 致命错误:缺少volatile关键字
    private static Singleton instance;
    
    // 私有构造方法
    private Singleton() {}
    
    public static Singleton getInstance() {
        // 第一次检查(无锁)
        if (instance == null) {
            // 加锁
            synchronized (Singleton.class) {
                // 第二次检查(有锁)
                if (instance == null) {
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
}

✅ 正确写法(加volatile禁止指令重排序)

java 复制代码
public class Singleton {
    // 关键:加上volatile关键字
    private static volatile Singleton instance;
    
    private Singleton() {}
    
    public static Singleton getInstance() {
        if (instance == null) {
            synchronized (Singleton.class) {
                if (instance == null) {
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
}

三、最致命坑:指令重排序

对象创建的两种执行顺序

复制代码
┌───────────── 正常顺序(安全) ─────────────┐
│ 1. 分配对象内存空间                        │
│ 2. 初始化对象(执行构造方法)              │
│ 3. 将instance引用指向该内存地址            │
└───────────────────────────────────────────┘

┌───────────── 重排序后(危险) ─────────────┐
│ 1. 分配对象内存空间                        │
│ 3. 将instance引用指向该内存地址            │
│ 2. 初始化对象(执行构造方法)              │
└───────────────────────────────────────────┘

灾难发生完整流程

复制代码
线程A执行 instance = new Singleton()
        ↓
JVM发生指令重排序:先执行步骤3,后执行步骤2
        ↓
instance ≠ null,但对象内部数据全是默认值(未初始化)
        ↓
线程B执行第一次检查:发现instance≠null,直接返回
        ↓
线程B使用这个"半成品"对象 → 抛出NullPointerException
相关推荐
半个落月12 小时前
从递归到快速排序:用 JavaScript 把分治思想讲明白
javascript·算法·面试
Darling噜啦啦12 小时前
快速排序与递归思维:从分治策略到数组扁平化——面试必考算法全解析
面试·排序算法
plainGeekDev12 小时前
单例模式 → object 声明
android·java·kotlin
小月土星13 小时前
JavaScript 快速排序:从 pivot、双指针到分治思想
javascript·算法·面试
沉默王二13 小时前
Agent底层原理连问8道,从ReAct到记忆压缩,PaiCLI项目实战拆解
面试·agent·ai编程
小月土星13 小时前
JavaScript 递归入门:从 1 到 n 求和,再到数组扁平化
javascript·算法·面试
用户2986985301413 小时前
Java 实现 Word 文档文本与图片提取的方法
java·后端
蝎子莱莱爱打怪13 小时前
XZLL-IM干货系列 04|Netty 长连接实战:Pipeline 怎么排、心跳怎么跳、连接怎么管
后端·微服务·面试
SimonKing14 小时前
铁子,IntelliJ IDEA 2026.1.3来了,升不升?
java·后端·程序员