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

单例模式总结

单例模式本身是一种设计模式,是否线程安全取决于具体实现方式。以下是不同单例实现方式的线程安全性分析及优化方案:


一、线程不安全的单例实现

1. 懒汉式(未同步)
public class Singleton {private static Singleton instance;private Singleton() {}public static Singleton getInstance() {if (instance == null) {                // 线程A和B可能同时进入此判断instance = new Singleton();        // 导致创建多个实例}return instance;}
}

问题:多线程同时调用 getInstance() 会破坏单例性。


二、线程安全的单例实现

1. 饿汉式(线程安全)
public class Singleton {private static final Singleton instance = new Singleton();private Singleton() {}public static Singleton getInstance() {return instance;  // 类加载时已初始化,JVM保证线程安全}
}

特点

  • 优点:实现简单,无锁性能高
  • 缺点:无法延迟加载(即使未使用也会初始化)

2. 懒汉式(同步方法)
public class Singleton {private static Singleton instance;private Singleton() {}public static synchronized Singleton getInstance() {  // 方法级同步if (instance == null) {instance = new Singleton();}return instance;}
}

特点

  • 优点:实现简单,保证线程安全
  • 缺点:每次调用都加锁,性能差(锁粒度太大)

3. 双重检查锁定(DCL,线程安全)
public class Singleton {private static volatile Singleton instance;  // 必须用volatileprivate Singleton() {}public static Singleton getInstance() {if (instance == null) {                  // 第一次检查(无锁)synchronized (Singleton.class) {     // 加锁if (instance == null) {          // 第二次检查(有锁)instance = new Singleton();  // volatile禁止指令重排序}}}return instance;}
}

关键点

  • volatile 防止指令重排序(避免返回未初始化的对象)
  • 减少锁竞争(仅首次创建时同步)

4. 静态内部类(线程安全)
public class Singleton {private Singleton() {}private static class Holder {static final Singleton INSTANCE = new Singleton();  // 类加载时初始化}public static Singleton getInstance() {return Holder.INSTANCE;  // 首次调用时加载Holder类}
}

特点

  • 结合懒加载与无锁线程安全
  • JVM保证类加载过程的线程安全性

5. 枚举实现(最佳实践)
public enum Singleton {INSTANCE;  // 枚举实例天然单例且线程安全public void doSomething() {// 业务方法}
}

优势

  • 绝对防止反射攻击(枚举类无法通过反射创建实例)
  • 自动处理序列化与反序列化
  • 代码最简洁

三、线程安全单例的核心要点

实现方式线程安全延迟加载性能防反射防序列化
饿汉式⭐⭐⭐⭐
同步懒汉式
双重检查锁定⭐⭐⭐
静态内部类⭐⭐⭐⭐
枚举⭐⭐⭐⭐

四、常见面试陷阱

  1. 为什么DCL需要volatile?

    • 防止指令重排序:new Singleton() 的字节码分为三步(分配内存→初始化→引用赋值),不加volatile可能导致其他线程获取到未初始化的对象。
  2. 静态内部类如何保证线程安全?

    • JVM在类加载时(首次访问Holder.INSTANCE)会加锁(ClassLoader机制),保证初始化过程线程安全。
  3. 枚举单例为何能防反射?

    • 枚举类的构造方法在反射调用时会抛出IllegalArgumentException,源码中明确禁止反射创建枚举实例。

五、总结回答

“单例模式能否保证线程安全取决于具体实现方式:

  1. 线程不安全实现:如无同步的懒汉式
  2. 线程安全实现
    • 饿汉式(简单但无法延迟加载)
    • 双重检查锁定(需加volatile
    • 静态内部类(兼顾性能与懒加载)
    • 枚举(最佳实践,防反射/序列化攻击)
      实际开发中推荐使用枚举或静态内部类实现单例模式。”

相关文章:

  • Python训练营---Day35
  • 【笔记】OpenCV的学习(未完)
  • 开发者工具箱-鸿蒙颜色转换器开发笔记
  • React Native 拼音及拼音首字母搜索组件开发
  • LLM Tuning
  • OpenLayers 开发环境搭建
  • 每日c/c++题 备战蓝桥杯(洛谷P3382 三分法求极值详解)
  • 模型参数详细介绍--Profiler
  • Python如何赋能自动驾驶地图构建?从点云处理到实时导航
  • Redis持久化机制:AOF与RDB深度解析
  • 电机控制储备知识学习(五) 三项直流无刷电机(BLDC)学习(四)
  • NetSuite Emoji的应用
  • JavaScript 文件类型识别与状态图片返回系统
  • [AI]主流大模型、ChatGPTDeepseek、国内免费大模型API服务推荐(支持LangChain.js集成)
  • 开发者工具箱-鸿蒙DNS查询工具开发笔记
  • vue3 发票税率 计算
  • 汉字不仅是一种语言 还是当作艺术形式来展现
  • Python 基础语法速查手册:从入门到精通
  • 怎么开发一个网络协议模块(C语言框架)之(四) 信号量初始化
  • STM32上配置图像处理库时常见错误总结
  • 京东网站的建设与发展/seosem是指什么意思
  • 红桃免费观看电视剧高清斗破苍穹/贵港网站seo
  • 常见的网站首页布局有哪几种/crm系统网站
  • 大型网站频道的建设需多人协同开发/怎么样建一个网站
  • 个体户营业执照可以网站备案/百度快速收录seo工具软件
  • 可以做宣传的网站/企业seo整站优化方案