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

以下是以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)` 可能创建多个实例。
  1. 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()` 三步操作顺序正确),保证可见性。
  1. 静态内部类原理:JVM 保证类加载过程线程安全,且只有调用 `getInstance()` 时才会加载内部类并初始化实例。

四、单例模式的破坏与防御

  1. 反射攻击:
  • 通过反射调用私有构造器破坏单例。

  • 防御:在构造器中判断实例是否已存在,若存在则抛出异常。

  1. 序列化破坏:
  • 反序列化会创建新实例。

  • 防御:实现 `readResolve()` 方法返回单例实例。

  1. 克隆破坏:
  • 实现 `Cloneable` 接口可能导致克隆出新对象。

  • 防御:重写 `clone()` 方法返回单例实例或抛出异常。


五、应用场景与注意事项

  1. 适用场景:
  • 资源管理(如线程池、数据库连接池)。

  • 全局配置管理。

  • 日志系统。

  1. 注意事项:
  • 避免滥用单例(增加耦合性,破坏可测试性)。

  • 分布式系统中需使用分布式锁(如Redis)实现集群单例。

  • 有状态对象慎用单例,优先无状态设计。


六、面试加分项:进阶问题与扩展

  1. Spring 单例管理:
  • Spring 默认使用三级缓存 + 双重检查实现单例,类似 DCL 机制。
  1. C++11 中的单例:

```cpp

class Singleton {

public:

static Singleton& getInstance() {

static Singleton instance; // C++11 保证线程安全

return instance;

}

private:

Singleton() {}

};

```

  1. 为什么枚举单例最安全?
  • 枚举本质是单例的语法糖,JVM 保证枚举实例唯一性,天然防反射和序列化。

总结

  • 必掌握:实现方式(尤其 DCL、枚举)、线程安全、防御机制。

  • 推荐实践:优先使用静态内部类或枚举单例,兼顾安全与性能。

  • 面试技巧:结合具体场景(如日志系统、配置管理)说明单例的适用性与设计思路。


希望这份Markdown整理能帮助你系统性掌握单例模式的面试重点!

相关推荐
JH30738 小时前
SpringBoot 优雅处理金额格式化:拦截器+自定义注解方案
java·spring boot·spring
Coder_Boy_9 小时前
技术让开发更轻松的底层矛盾
java·大数据·数据库·人工智能·深度学习
invicinble10 小时前
对tomcat的提供的功能与底层拓扑结构与实现机制的理解
java·tomcat
较真的菜鸟10 小时前
使用ASM和agent监控属性变化
java
黎雁·泠崖10 小时前
【魔法森林冒险】5/14 Allen类(三):任务进度与状态管理
java·开发语言
qq_124987075311 小时前
基于SSM的动物保护系统的设计与实现(源码+论文+部署+安装)
java·数据库·spring boot·毕业设计·ssm·计算机毕业设计
Coder_Boy_11 小时前
基于SpringAI的在线考试系统-考试系统开发流程案例
java·数据库·人工智能·spring boot·后端
Mr_sun.11 小时前
Day06——权限认证-项目集成
java
瑶山11 小时前
Spring Cloud微服务搭建四、集成RocketMQ消息队列
java·spring cloud·微服务·rocketmq·dashboard
abluckyboy11 小时前
Java 实现求 n 的 n^n 次方的最后一位数字
java·python·算法