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

创建型模式5.单例模式

创建型模式

  1. 工厂方法模式(Factory Method Pattern)
  2. 抽象工厂模式(Abstract Factory Pattern)
  3. 建造者模式(Builder Pattern)
  4. 原型模式(Prototype Pattern)
  5. 单例模式(Singleton Pattern)

单例模式(Singleton Pattern) 是一种设计模式,旨在确保一个类只有一个实例,并提供一个全局访问点来访问该实例。单例模式可以用于管理全局的资源,确保整个系统中只有一个对象来处理特定任务。

单例模式的关键特性:

  1. 确保只有一个实例:单例模式确保在整个应用程序生命周期内,某个类只有一个实例。
  2. 提供全局访问点:通过公共静态方法,可以访问该唯一的实例。
  3. 延迟实例化:有时实例化时机会被延迟,只有在需要时才创建实例(懒汉式)。

1. 懒汉式(Lazy Initialization)

懒汉式单例模式在第一次使用时创建实例。它通常是线程不安全的,但可以通过同步来确保线程安全。

线程不安全的懒汉式:
public class Singleton {private static Singleton instance;private Singleton() {// 私有构造函数,防止外部实例化}public static Singleton getInstance() {if (instance == null) {instance = new Singleton();  // 延迟实例化}return instance;}
}

问题:

  • 上述代码线程不安全。如果多个线程同时访问 getInstance() 方法,在没有同步的情况下,可能会创建多个实例。

线程安全的懒汉式: 

public class Singleton {private static Singleton instance;private Singleton() {// 私有构造函数,防止外部实例化}public synchronized static Singleton getInstance() {if (instance == null) {instance = new Singleton();  // 延迟实例化}return instance;}
}

优点:

  • 只有在需要实例时才创建(延迟加载)。

缺点:

  • 使用 synchronized 来保证线程安全会影响性能,因为每次调用 getInstance() 都需要加锁。

2. 饿汉式(Eager Initialization)

饿汉式单例模式在类加载时立即创建实例。线程安全,且没有懒汉式的延迟加载问题,但不适用于需要在应用程序启动时创建对象的场景。

public class Singleton {// 静态实例在类加载时创建private static final Singleton instance = new Singleton();private Singleton() {// 私有构造函数,防止外部实例化}public static Singleton getInstance() {return instance;}
}

优点:

  • 实例在类加载时创建,因此线程安全,不需要额外的同步操作。
  • 不会有性能问题。

缺点:

  • 如果类加载时没有使用单例对象,可能会浪费资源。

3. 双重检查锁定(Double-Checked Locking)

双重检查锁定是一种改进的懒汉式单例模式,使用 synchronized 来确保线程安全,但仅在实例为空时加锁,避免了每次调用 getInstance() 都加锁的性能损失。

public class Singleton {private static volatile Singleton instance;private Singleton() {// 私有构造函数,防止外部实例化}public static Singleton getInstance() {if (instance == null) {synchronized (Singleton.class) {if (instance == null) {instance = new Singleton();  // 延迟实例化}}}return instance;}
}

优点:

  • 只有在实例为空时才进行加锁,减少了加锁的开销,提供了懒加载的优点,并且是线程安全的。

缺点:

  • 需要使用 volatile 关键字,确保内存可见性。

4. 静态内部类方式(推荐)

静态内部类方式是实现单例模式的最佳方法之一。它利用了类的加载机制,保证了线程安全和延迟加载。

public class Singleton {// 静态内部类,只有在调用 getInstance 时才会加载private static class SingletonHolder {private static final Singleton INSTANCE = new Singleton();}private Singleton() {// 私有构造函数,防止外部实例化}public static Singleton getInstance() {return SingletonHolder.INSTANCE;}
}

优点:

  • 线程安全,且实现简单。
  • 延迟加载,只有在调用 getInstance() 时才实例化对象。
  • 通过类的加载机制实现线程安全,不需要显式加锁。
  • 具体的解析参考这个文章:静态内部类为什么在不需要加锁就是线程安全的单例模式-CSDN博客

总结:

  1. 懒汉式(线程不安全): 延迟实例化,适合需要延迟加载的场景,但需要处理线程安全问题。
  2. 饿汉式(线程安全): 类加载时实例化,线程安全,简单,但可能会造成资源浪费。
  3. 双重检查锁定: 提供了懒加载和线程安全的优势,性能较高。
  4. 静态内部类方式(推荐): 最佳的单例实现方式,延迟加载,线程安全,且简单。

对于大多数情况,静态内部类方式是最推荐的实现方式,因为它利用了Java的类加载机制,在保证线程安全的同时也保持了延迟加载的优点。

http://www.lryc.cn/news/516719.html

相关文章:

  • 用户界面软件02
  • VTK 鼠标+键盘重构
  • go语言处理JSON数据详解
  • 基于gin一个还算比较优雅的controller实现
  • PDFMathTranslate: Star13.8k,一款基于AI的PDF文档全文双语翻译PDF文档全文双语翻译,保留格式神器,你应该需要它
  • Python编程实例-特征向量与特征值编程实现
  • Vue3-跨层组件通信Provide/Inject机制详解
  • Linux Jar包定时重启脚本,按最新时间的Jar包启动
  • HTML5实现好看的博客网站、通用大作业网页模板源码
  • 掌握RabbitMQ:全面知识点汇总与实践指南
  • go如何从入门进阶到高级
  • 在环境冲突情况下调整优先级以解决ROS Catkin构建中缺少模块的问题【ubuntu20.04】
  • github 个人主页配置
  • STM32-笔记30-编程实现esp8266联网功能
  • oscp备考 oscp系列——Kioptix Level 1靶场 古老的 Apache Vuln
  • 《机器学习》——随机森林
  • 指代消解:自然语言处理中的核心任务与技术进展
  • 记录一下Unity webgl cannot read properties of undefined reading apply 错误
  • 【C语言程序设计——选择结构程序设计】求阶跃函数的值(头歌实践教学平台习题)【合集】
  • unity 播放 序列帧图片 动画
  • HTML - <a>
  • Unity学习笔记(六)使用状态机重构角色移动、跳跃、冲刺
  • 【C++数据结构——树】二叉树的遍历算法(头歌教学实验平台习题) 【合集】
  • Android Telephony | 协议测试针对 test SIM attach network 的问题解决(3GPP TS 36523-1-i60)
  • jenkins入门3 --执行一个小demo
  • STM32传感器系列:GPS定位模块
  • 技术成长战略是什么?
  • 【前端】Vue3与Element Plus结合使用的超详细教程:从入门到精通
  • Linux 35.6 + JetPack v5.1.4之 pytorch升级
  • 旷视科技C++面试题及参考答案