登录
/
注册
首页
论坛
其它
首页
科技
业界
安全
程序
广播
Follow
关于
签到
每天签到奖励2-10圆
导读
排行榜
TG频道
发帖说明
登录
/
注册
账号
自动登录
找回密码
密码
登录
立即注册
搜索
搜索
关闭
CSDN热搜
程序园
精品问答
技术交流
资源下载
本版
帖子
用户
软件
问答
教程
代码
写记录
VIP申请
VIP网盘
网盘
联系我们
发帖说明
每日签到
道具
勋章
任务
淘帖
动态
分享
留言板
导读
设置
我的收藏
退出
腾讯QQ
微信登录
返回列表
首页
›
业界区
›
安全
›
Java单例模式:从实战到面试的深度解析
Java单例模式:从实战到面试的深度解析
[ 复制链接 ]
拼潦
2025-5-31 23:59:01
结论先行
饿汉式
:线程安全但可能造成资源浪费,推荐在初始化成本低的场景使用
懒汉式
:需要解决线程安全问题,推荐使用双重检查锁+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
单例
模式
实战
面试
相关帖子
鸿蒙应用开发从入门到实战(十八):组件编程思想之代码复用
C# 弃元模式:从语法糖到性能利器的深度解析
靠谱的程序员推荐阅读----->《阿里Java开发手册》
SpringCloud微服务实战——搭建企业级开发框架
三类代码协同模式,你要如何选?
全球知名的Java Web开发平台Vaadin上线慧都网!
CopyOnWriteArrayList 的故事--一起看看java原生的读写分离
鸿蒙应用开发从入门到实战(十九):样式复用案例
List之高效安全的 Java 列表深复制工具:ListCopyUtils 的设计与实践
接口测试全流程实战:从工具到架构的深度解析
vip免费申请,1年只需15美金$
回复
使用道具
举报
提升卡
置顶卡
沉默卡
喧嚣卡
变色卡
千斤顶
照妖镜
相关推荐
业界
鸿蒙应用开发从入门到实战(十八):组件编程思想之代码复用
0
648
系味
2025-10-10
业界
C# 弃元模式:从语法糖到性能利器的深度解析
0
25
南宫玉英
2025-10-10
业界
靠谱的程序员推荐阅读----->《阿里Java开发手册》
0
942
扈怀易
2025-10-10
安全
SpringCloud微服务实战——搭建企业级开发框架
0
128
茅香馨
2025-10-10
安全
三类代码协同模式,你要如何选?
0
301
跟尴
2025-10-10
业界
全球知名的Java Web开发平台Vaadin上线慧都网!
0
543
谲脾
2025-10-11
业界
CopyOnWriteArrayList 的故事--一起看看java原生的读写分离
0
723
广性
2025-10-11
业界
鸿蒙应用开发从入门到实战(十九):样式复用案例
0
328
赖娅闺
2025-10-11
业界
List之高效安全的 Java 列表深复制工具:ListCopyUtils 的设计与实践
0
534
萨瑞饨
2025-10-11
科技
接口测试全流程实战:从工具到架构的深度解析
0
156
泠邸
2025-10-11
高级模式
B
Color
Image
Link
Quote
Code
Smilies
您需要登录后才可以回帖
登录
|
立即注册
回复
本版积分规则
回帖并转播
回帖后跳转到最后一页
浏览过的版块
问答
业界
代码
签约作者
程序园优秀签约作者
发帖
拼潦
2025-5-31 23:59:01
关注
0
粉丝关注
23
主题发布
板块介绍填写区域,请于后台编辑
财富榜{圆}
anyue1937
9994888
dage888
999994
3934307807
993690
4
富账慕
10007
5
柴古香
9992
6
匝抽
9986
7
筒濂
9983
8
孙淼淼
9992
9
凌彦慧
9985
10
崔瑜然
9984
查看更多