0

0

Java单例模式怎么写 Java双重检查锁定DCL实现【面试】

星降

星降

发布时间:2026-03-02 09:29:16

|

673人浏览过

|

来源于php中文网

原创

直接用 synchronized 方法写单例不行,因为每次调用 getInstance() 都需加锁,即使实例已创建也会导致线程阻塞、性能下降;正确解法是 DCL(双重检查锁定)+ volatile,确保高并发下线程安全且高效。

java单例模式怎么写 java双重检查锁定dcl实现【面试】

为什么直接用 synchronized 方法写单例不行

因为每次 getInstance() 都要加锁,性能差——哪怕实例已经创建好了,还要排队等锁。面试官想看的不是“能跑”,而是“高并发下既安全又高效”的解法。

常见错误现象:
- 没加 volatile,导致指令重排序,其他线程可能拿到未初始化完成的对象(JVM 会把对象分配内存 → 调构造函数 → 赋值给静态变量 这三步重排)
- 把 synchronized 加在方法上,或只锁住整个 if 块但没做二次检查
- 构造函数没设为 private,破坏单例语义

DCL 实现必须带 volatile 的完整写法

这是目前 JDK 6+ 最稳妥的懒汉式单例写法,兼顾线程安全与性能:

public class Singleton {
    private static volatile Singleton instance;
<pre class="brush:php;toolbar:false;">private Singleton() {} // 防止反射攻击可加 throw new RuntimeException()

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

}

关键点:
- volatile 禁止重排序 + 保证可见性
- 第一次判空(避免不必要的同步)
- 进入同步块后再判空(防止多线程同时通过第一层判断后重复创建)
- synchronized 锁的是类对象,不是 this,确保全局唯一锁

为什么不能用 static 内部类替代 DCL

可以,而且更简洁、无锁、天然线程安全,但面试问 DCL 就是在考你对内存模型和锁优化的理解,不是比谁写得短。

立即学习Java免费学习笔记(深入)”;

通义灵码
通义灵码

阿里云出品的一款基于通义大模型的智能编码辅助工具,提供代码智能生成、研发智能问答能力

下载

static 内部类方式是合法且推荐的生产写法,但它不涉及 volatile 和双重检查逻辑,答不出 DCL 的底层原因(如重排序、happens-before)容易被追问卡住。

使用场景差异:
- DCL:需要延迟初始化 + 明确控制同步粒度 + 面试考察 JVM 底层
- 静态内部类:更推荐用于大多数业务代码,加载时机由类加载器保证,无需额外同步

面试最容易被揪住的三个细节

这几个点一错,基本就暴露没真写过/调过并发代码:

  • volatile 缺失 —— 不是“好像能跑”,是“一定有概率出 bug”,尤其在低版本 JDK 或特定 CPU 架构下
  • synchronized 锁在 this 或实例上 —— 单例还没创建,哪来的 this?编译都过不去
  • 没处理反射、反序列化、克隆攻击 —— 至少提一句“生产中需配合 readResolve() 和私有构造函数抛异常”

复杂点不在写几行代码,而在理解为什么那行 volatile 不可省,以及它和 synchronized 在内存屏障层面怎么协作。漏掉这个,DCL 就只是个模板,不是解决方案。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

838

2023.08.22

c++中volatile关键字的作用
c++中volatile关键字的作用

本专题整合了c++中volatile关键字的相关内容,阅读专题下面的文章了解更多详细内容。

72

2025.10.23

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

723

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

372

2025.12.24

java多线程相关教程合集
java多线程相关教程合集

本专题整合了java多线程相关教程,阅读专题下面的文章了解更多详细内容。

27

2026.01.21

C++多线程相关合集
C++多线程相关合集

本专题整合了C++多线程相关教程,阅读专题下面的的文章了解更多详细内容。

25

2026.01.21

C# 多线程与异步编程
C# 多线程与异步编程

本专题深入讲解 C# 中多线程与异步编程的核心概念与实战技巧,包括线程池管理、Task 类的使用、async/await 异步编程模式、并发控制与线程同步、死锁与竞态条件的解决方案。通过实际项目,帮助开发者掌握 如何在 C# 中构建高并发、低延迟的异步系统,提升应用性能和响应速度。

102

2026.02.06

Golang 测试体系与代码质量保障:工程级可靠性建设
Golang 测试体系与代码质量保障:工程级可靠性建设

Go语言测试体系与代码质量保障聚焦于构建工程级可靠性系统。本专题深入解析Go的测试工具链(如go test)、单元测试、集成测试及端到端测试实践,结合代码覆盖率分析、静态代码扫描(如go vet)和动态分析工具,建立全链路质量监控机制。通过自动化测试框架、持续集成(CI)流水线配置及代码审查规范,实现测试用例管理、缺陷追踪与质量门禁控制,确保代码健壮性与可维护性,为高可靠性工程系统提供质量保障。

20

2026.02.28

Golang 工程化架构设计:可维护与可演进系统构建
Golang 工程化架构设计:可维护与可演进系统构建

Go语言工程化架构设计专注于构建高可维护性、可演进的企业级系统。本专题深入探讨Go项目的目录结构设计、模块划分、依赖管理等核心架构原则,涵盖微服务架构、领域驱动设计(DDD)在Go中的实践应用。通过实战案例解析接口抽象、错误处理、配置管理、日志监控等关键工程化技术,帮助开发者掌握构建稳定、可扩展Go应用的最佳实践方法。

15

2026.02.28

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4万人学习

C# 教程
C# 教程

共94课时 | 10.5万人学习

Java 教程
Java 教程

共578课时 | 74.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号