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
相关推荐
在繁华处1 小时前
Java从零到熟练(四):面向对象基础
java·开发语言
小江的记录本3 小时前
【JVM虚拟机】堆内存分代模型:年轻代(Eden+Survivor)、老年代、元空间Metaspace(附《思维导图》+《面试高频考点清单》)
java·前端·jvm·后端·python·spring·面试
在繁华处3 小时前
Java从零到熟练(三):流程控制
java·开发语言·python
唐青枫3 小时前
Java Optional 实战指南:优雅处理空值与链式转换
java
一起学开源3 小时前
一文读懂 ReAct 范式:让 AI Agent 真正学会“思考+行动“
java·javascript·react.js·ecmascript·react·alibaba·智能体开发
逍遥德4 小时前
MQTT教程详解-04.SpringBoot集成MQTT(告别手动控制)
java·spring boot·物联网·中间件·iot·iotdb
语戚4 小时前
力扣 3161. 块放置查询:线段树解法(Java 实现)
java·算法·leetcode·面试·线段树·力扣·
天天进步20154 小时前
Python全栈项目实战:从零构建校园心理健康咨询平台
面试·职场和发展
我命由我123455 小时前
Android 开发问题:MlKitException: An internal error occurred during initialization.
android·java·java-ee·android jetpack·android-studio·androidx·android runtime