news 2026/6/15 10:13:48

懒加载单例模式中DCL方式和原理解析

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
懒加载单例模式中DCL方式和原理解析

一、DCL 是什么?

DCL(Double Check Lock,双重检查锁)是 Java 中懒加载单例模式的高性能实现方案,核心思路是:

  • 第一次检查:无锁判断实例是否已初始化,避免每次调用都加锁(提升性能);
  • 加锁:保证多线程下只有一个线程能进入初始化逻辑;
  • 第二次检查:防止多个线程等待锁后重复初始化实例。

它解决了传统 “懒汉式(同步方法)” 每次调用都加锁的性能问题,同时保证线程安全。

二、DCL 完整实现代码(标准写法)

public class SingletonDCL { // 1. 必须加 volatile 关键字! private static volatile SingletonDCL INSTANCE; // 2. 私有构造器,禁止外部实例化 private SingletonDCL() {} // 3. 双重检查锁的核心方法 public static SingletonDCL getInstance() { // 第一次检查:无锁,快速判断实例是否已存在 // 若已存在,直接返回,无需加锁,提升高并发性能 if (INSTANCE == null) { // 加锁:保证同一时间只有一个线程能进入初始化逻辑 synchronized (SingletonDCL.class) { // 第二次检查:防止多个线程等待锁后重复创建实例 if (INSTANCE == null) { // 初始化实例 INSTANCE = new SingletonDCL(); } } } return INSTANCE; } }

三、DCL 核心逻辑拆解(为什么要 “双重检查”)

假设高并发场景下有 3 个线程(T1、T2、T3)同时调用getInstance()

  1. T1 先执行:第一次检查INSTANCE == null为 true,进入加锁逻辑,第二次检查仍为 true,执行INSTANCE = new SingletonDCL(),初始化完成后释放锁;
  2. T2 随后执行:第一次检查INSTANCE已不为 null,直接返回实例,无需加锁;
  3. T3 与 T1 同时执行:T3 先通过第一次检查(此时 T1 还未完成初始化),等待 T1 释放锁后进入加锁逻辑,第二次检查发现INSTANCE已被 T1 初始化,直接返回,避免重复创建。

如果去掉 “第二次检查”,T3 会在 T1 释放锁后重新创建实例,导致单例失效。

四、为什么必须加volatile?(面试必考)

这是 DCL 最核心的坑点!INSTANCE = new SingletonDCL()看似一行代码,实际 JVM 会拆分为 3 步执行:

1. 分配内存空间(给 SingletonDCL 实例); 2. 初始化实例(执行构造器逻辑,给成员变量赋值); 3. 将 INSTANCE 引用指向分配的内存空间(此时 INSTANCE 不再为 null)。

JVM 为了优化性能,可能会对这 3 步进行指令重排(比如重排为 1→3→2),导致问题:

  • T1 执行时,JVM 先执行 1→3(INSTANCE 不为 null,但实例还未初始化);
  • T2 此时第一次检查INSTANCE != null,直接返回这个 “半初始化” 的实例;
  • T2 调用实例的方法时,会因实例未初始化完成抛出空指针或逻辑错误。

volatile关键字的核心作用:禁止 JVM 对指令重排,保证 1→2→3 的执行顺序,确保其他线程看到的INSTANCE要么是 null,要么是完全初始化的实例。

五、DCL 的常见误区(避坑)

错误写法问题说明
去掉volatile可能拿到 “半初始化” 实例,线程安全失效
去掉第二次检查多线程等待锁后重复创建实例,单例失效
同步代码块锁对象错误(比如锁thisstatic方法中this不存在,且锁对象不唯一,线程安全失效
INSTANCE定义为finalfinal变量必须初始化,无法实现懒加载

六、DCL 的优缺点

优点缺点

1. 懒加载:实例仅在第一次调用时初始化,节省内存;

2. 高性能:仅初始化时加锁,后续调用无锁;

3. 线程安全:双重检查 + volatile 保证单例唯一性

1. 实现稍复杂,新手易遗漏 volatile;

2. 无法解决 “反射 / 序列化破坏单例” 的问题(需额外处理);

3. 在早期 JDK(1.4 及之前)中,volatile 实现有缺陷,DCL 可能失效(现代 JDK 已修复)

七、DCL 的适用场景

  • 高并发场景下的懒加载单例(比如工具类、连接池、配置中心实例);
  • 对内存占用敏感,需要延迟初始化的场景;
  • 追求高性能,不希望每次调用都加锁的场景。

总结

  1. DCL 是 “懒加载 + 高性能 + 线程安全” 的单例实现方案,核心是 “两次检查 + 加锁 + volatile”;
  2. volatile是 DCL 的关键,用于禁止指令重排,避免拿到半初始化实例;
  3. 第二次检查不可省略,否则多线程下会重复创建实例;
  4. 现代 JDK(1.5+)中 DCL 是安全的,是生产环境中最常用的单例实现方式之一。
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/15 14:42:57

专科生也能用!学生热捧的AI论文软件 —— 千笔AI

你是否曾为论文选题发愁?是否在写到一半时突然卡文,毫无头绪?又或是反复修改却总达不到老师要求?对于专科生来说,论文写作不仅是学术任务,更是一场心理挑战。面对查重率、格式规范、文献查找等难题&#xf…

作者头像 李华
网站建设 2026/6/14 2:11:15

一文读懂VR/AR/MR:小白也能分清的虚实交互技术

一文读懂 VR / AR / MR(附 XR)——小白也能分清的虚实交互技术(2026 年最新视角) 2026 年了,头显设备已经从“玩具”变成了部分人的生产力工具和娱乐刚需,但很多人一听到 VR、AR、MR、XR 还是会懵。 其实它…

作者头像 李华
网站建设 2026/6/15 12:18:45

根据Excel 模板生成excel(java)

重阳,需求很常见:根据一个已设计好的 Excel 模板(含样式、公式、合并单元格、表头、页眉页脚等),往里面填充动态数据,然后生成新的 Excel 文件。 2026 年主流的三种方案对比(基于当前社区使用情…

作者头像 李华
网站建设 2026/6/15 13:18:45

降重≠换词!虎贲等考 AI:双重净化 AIGC 痕迹,查重率直降 25%+

“论文降重改到崩溃,重复率还是飙到 30%”“AI 写的初稿被导师标记‘机器感过重’”“降重后语句不通,学术逻辑全乱”—— 这是当下毕业生写论文时最头疼的两大痛点。单纯的同义词替换早已过时,既过不了查重关,还可能因 AIGC 痕迹…

作者头像 李华