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

单例模式的设计与实现

文章目录

  • 一、不安全的单例模式
  • 二、线程安全的单例模式
    • 1. 同步方法
    • 2. 静态内部类
    • 3. 枚举:最安全的实现方式

单例模式可能是我们在开发中用得最多的设计模式之一,但要在多线程环境下正确实现单例模式却不是那么简单。今天我们就来看看如何正确地实现线程安全的单例模式。

一、不安全的单例模式

在单线程环境下,实现单例模式很简单,但在多线程环境下就不安全了,可能会出现多个线程同时创建实例的情况,这就违背了单例模式的初衷。

// 这种实现在多线程环境下是不安全的
public class Singleton {private static Singleton instance;private Singleton () {}public static Singleton getInstance() {if (instance == null) {// 如果多个线程同时执行到这里,就可能创建多个实例instance = new Singleton ();}return instance;}
}

如果线程A和线程B同时调用getInstance方法,都发现instance为null,那么它们可能会同时创建实例,这就破坏了单例的约束。

二、线程安全的单例模式

1. 同步方法

最直接的解决方案就是给getInstance方法加上synchronized关键字:

public class Singleton {private static Singleton instance;private Singleton () {}public static synchronized Singleton getInstance() {if (instance == null) {instance = new Singleton ();}return instance;}
}

这种方式确实能保证线程安全,但是有个明显的性能问题:每次调用getInstance都需要获取锁,即使实例已经创建好了。在高并发场景下,这会成为一个性能瓶颈。

2. 静态内部类

public class StaticInnerClassSingleton {private StaticInnerClassSingleton() {}private static class SingletonHolder {private static final StaticInnerClassSingleton INSTANCE = new StaticInnerClassSingleton();}public static StaticInnerClassSingleton getInstance() {return SingletonHolder.INSTANCE;}
}

这种实现利用了JVM的类加载机制来保证线程安全。静态内部类SingletonHolder只有在被引用时才会被加载,而类的加载过程是线程安全的,JVM会保证一个类只被加载一次。
但是普通的单例实现可能会被反射攻击破坏:

// 反射攻击示例
Constructor<StaticInnerClassSingleton> constructor = StaticInnerClassSingleton.class.getDeclaredConstructor();
constructor.setAccessible(true);
StaticInnerClassSingleton instance1 = constructor.newInstance();
StaticInnerClassSingleton instance2 = StaticInnerClassSingleton.getInstance();
// instance1 和 instance2 是不同的对象

3. 枚举:最安全的实现方式

在《Effective Java》中推荐了使用枚举来实现单例模式:

public enum EnumSingleton {INSTANCE;public void doSomething() {// 业务方法}
}// 使用方式
EnumSingleton.INSTANCE.doSomething();

枚举实现单例有几个独特的优势

  1. 枚举的实例创建是线程安全的,JVM会保证枚举实例只被创建一次。

  2. 但是枚举天然防止反射攻击,因为JVM不允许通过反射创建枚举实例。

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

相关文章:

  • Helm常用命令大全(2025最新版)
  • 马拉车(Manacher)算法
  • 自动化立体仓库堆垛机控制系统上报堆垛机状态 FC5
  • PCA主成分分析
  • js (bom)
  • [论文阅读] 软件工程 | 用模糊逻辑“解锁”项目成功:告别非黑即白的评估时代
  • MybatisPlus-08.核心功能-IService开发基础业务接口
  • CAN通信驱动开发注意事项
  • 【工具自荐】配置文件管理器:支持本地与远程配置文件的统一管理、编辑、刷新
  • TCP/IP 哲学:端到端的 Postel 定律
  • 从单个神经元到数字识别神经网络的演变
  • 【黄山派-SF32LB52】—硬件原理图学习笔记
  • Java从入门到精通:全面学习路线指南
  • 阿里云ssl证书自动安装及续订(acme)
  • Python之--元组
  • 7月19日 台风“韦帕“强势逼近:一场与时间赛跑的防御战
  • 回溯算法(Backtracking Algorithm)
  • day056-Dockerfile案例与Docker Compose
  • docker run快速启动一个容器
  • C++ :vector的模拟
  • 第J8周打卡
  • 【Linux】LVS(Linux virual server)环境搭建
  • uni-app 应用、页面、组件生命周期
  • Python+大模型 day02
  • 1.2M 小工具! 解决 Windows 系统疑难杂症
  • MVP 最小可行产品
  • 《Electron应用性能深耕:资源加载与内存治理的进阶路径》
  • 【51单片机学习】LED、独立按键
  • 一站式PDF转Markdown解决方案PDF3MD
  • Python技术题2