当前位置: 首页 > news >正文

设计模式面试之单例模式常问知识点

以下是以Markdown格式整理的单例模式在面试中的常问知识点,包含结构化的分点、代码示例和关键解析:
设计模式面试之单例模式常问知识点
---
一、单例模式的定义与核心目标
- 定义:确保一个类在JVM中仅存在一个实例,并提供一个全局访问点。
- 核心要素:
1. 私有化构造方法(防止外部直接实例化)。
2. 持有私有静态实例引用。
3. 提供静态公有方法获取实例(如 `getInstance()`)。
---
二、单例模式的常见实现方式与对比
| 类型       | 特点                                                                 | 线程安全 | 适用场景                     |
|----------------|--------------------------------------------------------------------------|--------------|----------------------------------|
| 饿汉式     | 类加载时立即初始化实例(静态变量)。                                     | ✅ 天然安全   | 对象小、频繁访问且无需延迟加载。 |
| 懒汉式(基础) | 延迟加载,首次调用时创建实例。                                       | ❌ 非安全     | 需延迟初始化的场景。             |
| 懒汉式(线程安全) | 加锁(`synchronized` 方法或代码块)解决线程安全问题。               | ✅ 安全       | 但性能较低。                     |
| 双重检查锁(DCL) | 懒加载 + 两次判空 + `synchronized` + `volatile` 防止指令重排。         | ✅ 安全高效   | 高并发下的延迟加载。             |
| 静态内部类 | 利用JVM类加载机制保证线程安全,延迟加载。                               | ✅ 安全       | 推荐使用,兼顾安全与延迟加载。   |
| 枚举       | 天然线程安全,防反射、序列化攻击,简洁高效。                             | ✅ 安全       | 最推荐实现方式。                 |
示例代码:
```java
// 饿汉式
public class Singleton {
private static final Singleton INSTANCE = new Singleton();
private Singleton() {}
public static Singleton getInstance() { return INSTANCE; }
}
// 静态内部类
public class Singleton {
private static class Holder {
static final Singleton INSTANCE = new Singleton();
}
private Singleton() {}
public static Singleton getInstance() { return Holder.INSTANCE; }
}
// 枚举(最推荐)
public enum Singleton {
INSTANCE;
public void someMethod() {... }
}
```
---
三、线程安全与性能优化
1. 懒汉式的线程问题:
- 多线程同时进入 `if (instance == null)` 可能创建多个实例。
2. DCL的实现与 `volatile` 必要性:
```java
public class Singleton {
private volatile static Singleton instance;
private Singleton() {}
public static Singleton getInstance() {
if (instance == null) {
synchronized (Singleton.class) {
if (instance == null) {
instance = new Singleton(); // 可能指令重排:分配内存 → 赋值引用 → 初始化
}
}
}
return instance;
}
}
```
- `volatile` 作用:禁止指令重排(确保 `new Singleton()` 三步操作顺序正确),保证可见性。
3. 静态内部类原理:JVM 保证类加载过程线程安全,且只有调用 `getInstance()` 时才会加载内部类并初始化实例。
---
四、单例模式的破坏与防御
1. 反射攻击:
- 通过反射调用私有构造器破坏单例。
- 防御:在构造器中判断实例是否已存在,若存在则抛出异常。
2. 序列化破坏:
- 反序列化会创建新实例。
- 防御:实现 `readResolve()` 方法返回单例实例。
3. 克隆破坏:
- 实现 `Cloneable` 接口可能导致克隆出新对象。
- 防御:重写 `clone()` 方法返回单例实例或抛出异常。
---
五、应用场景与注意事项
1. 适用场景:
- 资源管理(如线程池、数据库连接池)。
- 全局配置管理。
- 日志系统。
2. 注意事项:
- 避免滥用单例(增加耦合性,破坏可测试性)。
- 分布式系统中需使用分布式锁(如Redis)实现集群单例。
- 有状态对象慎用单例,优先无状态设计。
---
六、面试加分项:进阶问题与扩展
1. Spring 单例管理:
- Spring 默认使用三级缓存 + 双重检查实现单例,类似 DCL 机制。
2. C++11 中的单例:
```cpp
class Singleton {
public:
static Singleton& getInstance() {
static Singleton instance; // C++11 保证线程安全
return instance;
}
private:
Singleton() {}
};
```
3. 为什么枚举单例最安全?
- 枚举本质是单例的语法糖,JVM 保证枚举实例唯一性,天然防反射和序列化。
---
总结
- 必掌握:实现方式(尤其 DCL、枚举)、线程安全、防御机制。
- 推荐实践:优先使用静态内部类或枚举单例,兼顾安全与性能。
- 面试技巧:结合具体场景(如日志系统、配置管理)说明单例的适用性与设计思路。
---
希望这份Markdown整理能帮助你系统性掌握单例模式的面试重点!

 

http://www.dtcms.com/a/390755.html

相关文章:

  • 深入解析 MySQL 元数据锁 (MDL) 与 SHOW PROCESSLIST 实战
  • 能不能写一个可以在linux使用的类nano编辑器
  • Rocky10 使用kubeadm部署K8s v1.34 一主两从
  • 深入理解Buffer:数据世界的“蓄水池“
  • 通义万相开源 Wan2.2-S2V-14B,实现图片+音频生成电影级数字人视频
  • windows c++环境 使用VScdoe配置opencv
  • JVM(四)-- 对象的实例化内存布局和直接内存
  • G1垃圾回收器的优势
  • 内存分配策略
  • Python采集Tik Tok视频详情,Tik TokAPI接口(json数据返回)
  • 实时通信技术大比拼:长轮询、短轮询、WebSocket 与 SSE 深度解析及实战指南
  • ICML 2025|图像如何与激光雷达对齐并互补?迈向协调的多模态3D全景分割
  • 基于Web的3D工程应用图形引擎——HOOPS Communicator技术解析
  • 【每日一问】运放的失调电压是什么?对于电路有何影响?
  • 【轨物方案】轨物科技新型储能管理系统:以AIoT技术驱动储能资产全生命周期价值最大化
  • 线性回归 vs 逻辑回归:从原理到实战的全面对比
  • HashMap的底层原理
  • 股指期货超短线如何操作?
  • 【洛谷】算法竞赛中的树结构:形式、存储与遍历全解析
  • 育苗盘补苗路径规划研究
  • API Gateway :API网关组件
  • conda激活虚拟环境
  • 重构大qmt通达信板块预警自动交易系统--读取通达信成分股
  • 25.9.19 Spring AOP
  • d38: PostgreSQL 简单入门与 Vue3 动态路由实现
  • No006:订阅化时间管理——迈向个性化、生态化的AI服务模式
  • 微服务-sentinel的理论与集成springcloud
  • C++学习:哈希表unordered_set/unordered_map的封装
  • 圆柱永磁体磁场及梯度快速计算与可视化程序
  • 种群演化优化算法:原理与Python实现