Java单例模式:从实战到面试的深度解析
结论先行
- 饿汉式:线程安全但可能造成资源浪费,推荐在初始化成本低的场景使用
- 懒汉式:需要解决线程安全问题,推荐使用双重检查锁+volatile优化
- 静态内部类:最佳实践方案,完美平衡延迟加载与线程安全
- 枚举单例:JDK1.5+推荐方案,天然防反射/序列化破坏
- 实际开发中优先选择枚举或静态内部类实现
文章持续更新,可以微信搜一搜「 半个脑袋儿 」第一时间阅读
一、核心实现方式
1. 饿汉式
class ClassA {
private static final ClassA INSTANCE = new ClassA();
public static ClassA getInstance() {
return INSTANCE;
}
private ClassA() {} // 防止反射创建
}
特点:
- 类加载时立即初始化(可能造成资源浪费)
- 天然线程安全
- 需处理反射攻击(添加私有构造器判空逻辑)
2. 双重检查锁懒汉式
class ClassB {
private static volatile ClassB instance;
public static ClassB getInstance() {
if (instance == null) { // 第一次检查
synchronized (ClassB.class) { // 同步锁
if (instance == null) { // 第二次检查
instance = new ClassB();
}
}
}
return instance;
}
private ClassB() {}
}
关键点:
volatile防止指令重排序(JDK5+的JMM修复)- 两次null检查确保性能与线程安全
- 仍可能被反射破坏单例
3. 静态内部类
class ClassC {
private static class Holder {
static final ClassC INSTANCE = new ClassC();
}
public static ClassC getInstance() {
return Holder.INSTANCE;
}
private ClassC() {}
}
优势:
- 利用类加载机制保证线程安全
- 实现延迟加载(调用getInstance时才会初始化)
- 代码简洁无锁
4. 枚举式
enum EnumSingleton {
INSTANCE;
public void businessMethod() {
// 业务方法
}
}
绝对优势:
- 天生防反射攻击(枚举类没有构造器)
- 自动处理序列化/反序列化
- 代码极度简洁
二、实战应用场景
1. 配置管理类
public enum ConfigManager {
INSTANCE;
private Properties props = new Properties();
ConfigManager() {
try(InputStream is = getClass().getResourceAsStream("/app.properties")) {
props.load(is);
}
}
public String getProperty(String key) {
return props.getProperty(key);
}
}
2. 数据库连接池
public class ConnectionPool {
private static final int MAX_SIZE = 100;
private BlockingQueue<Connection> pool = new ArrayBlockingQueue<>(MAX_SIZE);
private static class Holder {
static final ConnectionPool INSTANCE = new ConnectionPool();
}
private ConnectionPool() {
// 初始化连接池
}
public static ConnectionPool getInstance() {
return Holder.INSTANCE;
}
public Connection getConnection() throws InterruptedException {
return pool.take();
}
}
3. Spring中的单例
- Spring默认的Bean作用域就是单例
- 通过IOC容器管理生命周期
- 与设计模式单例的区别:每个容器对应一个实例
三、高频面试题解析
Q1:DCL(双重检查锁)为什么要加volatile?
答:防止指令重排序导致返回未初始化完成的对象。new操作不是原子操作,分为:
- 分配内存空间
- 初始化对象
- 将引用指向内存地址
不加volatile可能导致步骤2和3重排序,其他线程可能拿到未初始化完成的对象。
Q2:如何防止反射攻击?
private ClassC() {
if (Holder.INSTANCE != null) {
throw new RuntimeException("禁止反射创建!");
}
}
Q3:枚举单例如何防止反射?
- 枚举类的构造方法由JVM特殊处理
- 反射newInstance方法会直接抛出异常
Q4:单例对象什么时候会被回收?
- 只有当加载该类的ClassLoader被回收时才会被回收
- 一般情况(使用系统类加载器)会与JVM生命周期一致
Q5:单例模式的优缺点?
优点:
- 内存中只有一个实例,减少内存开销
- 避免对资源的多重占用
缺点:
- 违背单一职责原则(既要管理实例又要处理业务)
- 扩展困难(需要修改源码)
- 测试困难(全局状态难以隔离)
Java单例模式:从实战到面试的深度解析的更多相关文章
- java内存分配和String类型的深度解析
[尊重原创文章出自:http://my.oschina.net/xiaohui249/blog/170013] 摘要 从整体上介绍java内存的概念.构成以及分配机制,在此基础上深度解析java中的S ...
- 【转】java内存分配和String类型的深度解析
一.引题 在java语言的所有数据类型中,String类型是比较特殊的一种类型,同时也是面试的时候经常被问到的一个知识点,本文结合java内存分配深度分析关于String的许多令人迷惑的问题.下面是本 ...
- Java字符串池(String Pool)深度解析
版权声明:本文为博主原创文章,转载请注明出处,欢迎交流学习! 在工作中,String类是我们使用频率非常高的一种对象类型.JVM为了提升性能和减少内存开销,避免字符串的重复创建,其维护了一块特殊的内存 ...
- Java 的抽象特性:抽象类与接口深度解析
要点: 抽象类 接口 抽象类与接口的差别 一. 抽象 对于面向对象编程来说,抽象是它的四大特征之中的一个. 在Java中,能够通过两种形式来体现OOP的抽象:接口和抽象类. 接口和抽象类为我们提供了一 ...
- Java字符串池(String Pool)深度解析(转)
出自 http://www.cnblogs.com/fangfuhai/p/5500065.html 在工作中,String类是我们使用频率非常高的一种对象类型.JVM为了提升性能和减少内存开销,避 ...
- 重学 Java 设计模式:实战单例模式
作者:小傅哥 博客:https://bugstack.cn 沉淀.分享.成长,让自己和他人都能有所收获! 一.前言 5个创建型模式的最后一个 在设计模式中按照不同的处理方式共包含三大类:创建型模式.结 ...
- 重学 Java 设计模式:实战迭代器模式「模拟公司组织架构树结构关系,深度迭代遍历人员信息输出场景」
作者:小傅哥 博客:https://bugstack.cn - 原创系列专题文章 沉淀.分享.成长,让自己和他人都能有所收获! 一.前言 相信相信的力量! 从懵懂的少年,到拿起键盘,可以写一个Hell ...
- 【深入】java 单例模式(转)
[深入]java 单例模式 关于单例模式的文章,其实网上早就已经泛滥了.但一个小小的单例,里面却是有着许多的变化.网上的文章大多也是提到了其中的一个或几个点,很少有比较全面且脉络清晰的文章,于是,我便 ...
- Java软件系统功能设计实战训练视频教程
Java软件系统功能设计实战训练视频教程 第01节课:整体课程介绍和杂项介绍第02节课:软件功能设计常见理念和方法第03节课:关于软件设计的一些思考第04节课:第一周作业的业务和相应模式:综合应用简单 ...
- jvm--深入理解java虚拟机 精华总结(面试)(转)
深入理解java虚拟机 精华总结(面试)(转) 原文地址:http://www.cnblogs.com/prayers/p/5515245.html 一.运行时数据区域 3 1.1 程序计数器 3 1 ...
随机推荐
- 如何让领导轻松在本地查看Allure报告
如何让领导轻松在本地查看Allure报告 问题描述 当我们把精心生成的Allure报告原始文件发送给领导后,领导直接打开index.html文件时,页面却一直处于加载状态,无法显示数据. 通过F12开 ...
- SQL Server统计信息更新会被阻塞或引起会话阻塞吗?
在SQL Server数据库中,统计信息更新(UPDATE STATISTICS)会被其它会话阻塞吗?统计信息更新(UPDATE STATISTICS)会引起其它会话阻塞吗?在回答这两个问题前,我们必 ...
- FLink12--KeyByReduceApp
一.依赖 参考博文:https://www.cnblogs.com/robots2/p/16048648.html 二.代码 package net.xdclass.class9; import ja ...
- FreeSql学习笔记——2.插入
前言 由于还没有表结构,就先从新增开始,插入一些数据后才好做查询.修改.删除操作. 初始化 前面注入FreeSql时设置过自动同步表结构,那么就不用管数据库了,只需要在项目中定义实体,就会自动生成表结 ...
- xcode 12.3 mac m1
- vue路由$router.push()的三种传参方式
- Deepseek学习随笔(6)--- API 开发与自动化
获取 API Key 要开始使用 DeepSeek 的 API,你首先需要获取 API Key: 登录 DeepSeek 控制台 . 进入 API 管理 页面,生成 API Key. API 调用示例 ...
- QT5笔记:17. QComboBox和QPlainTextEdit
例子 #include "widget.h" #include "ui_widget.h" #include <QTextBlock> Widget ...
- 八米云-N1盒子、S905系列机顶盒等设备-小白保姆式超详细刷机教程
线刷准备 这里以魔百盒CM211-1为例,本次刷机用到的零碎工具比较多,不过都是常见刚需设备,大家可以按照清单核对一下. 目前只支持S905 L3.L3a.L2 系列的各种盒子 机顶盒本体 电脑一台 ...
- JavaGUI - [04] BoxLayout
题记部分 一.简介 为了简化开发,Swing引入了一个新的布局管理器:BoxLayout.BoxLayout可以在垂直和水平两个方向上摆放GUI组件,BoxLayout提供了如下一个简单的构造器: ...