【再谈设计模式】单例模式~唯一性的守护者

一、引言

        在软件工程中,软件开发,设计模式是提高代码复用性和可维护性的有效工具。单例模式(Singleton Pattern)作为一种创建型设计模式,旨在确保一个类只有一个实例,并提供对该实例的全局访问。这一模式在许多场景中都显得尤为重要,尤其是在需要共享资源或协调系统状态时。

二、定义与描述

        单例模式的核心思想是限制一个类仅能实例化一次,并提供一个全局访问点。这样可以避免重复实例化导致的资源浪费和状态不一致的问题。

结构

单例模式通常包含以下几个组成部分:

  • 私有构造函数:防止外部类直接实例化。
  • 一个静态变量:持有唯一的实例。
  • 一个静态方法:用于获取该实例。

三、抽象背景

        单例模式的背景可以追溯到多个设计需求的实际应用。例如,当一个应用程序需要使用一个共享的配置文件、数据库连接或共享资源时,单例模式便成为一个很好的选择。在这种情况下,程序不能允许多个实例同时存在,以免造成数据污染和不一致。

四、适用场景与现实问题解决

单例模式常用于以下场景:

  • 配置管理:在整个应用中共享配置文件或参数。
  • 日志管理:全局共享的日志记录器。
  • 数据库连接池:管理数据库连接的共享实例。

  • 线程池管理:确保线程池在整个应用中只有一个实例。

       通过应用单例模式,可以有效避免各类资源的误用和浪费,从而提升程序的性能和稳定性。旨在确保一个类只有一个实例,并提供一个全局访问点。为了让这个概念更易于理解,我们可以使用一个生活中的实际问题来说明。

生活例子:办公室的打印机

        在一个办公室工作,办公室里有一台打印机。现在我们来看看这台打印机在使用中的几种情况:

        唯一性:这个办公室只能有一台打印机。想象一下,如果有多台打印机,可能会导致很多问题,比如文件打印到错误的打印机、打印任务混乱等。因此,我们需要确保只有一台打印机。

        共享访问:办公室里的每个人都需要使用这台打印机,无论是打印文件还是复印资料。因此,我们需要一个简单的方法让每个人都能访问这台打印机,而不需要每次都去找。

        节省资源:维护多台打印机的成本和管理会耗费大量时间和精力。只有一台打印机可以集中管理,更加高效。

单例模式的类比

        在编程中,单例模式就像这台办公室的打印机一样。它确保一个类(例如打印机类)只有一个实例,并提供一个全局的访问点。在代码中实现单例模式通常涉及以下几个步骤:

  • 私有构造函数:防止其他地方直接创建新实例。
  • 静态变量:用于保存唯一的实例。
  • 公共静态方法:提供访问该实例的方式。

        通过这种方式,代码中需要使用打印机的地方,都会获取到同一个实例,避免了实例重复,确保了全局统一性。

        单例模式就像是确保办公室只有一台打印机,避免了资源的浪费和管理的复杂性,让每个人都可以方便地访问这台打印机。而在实际编程中,单例模式则确保了一个类的唯一性和资源的高效利用。

五、初衷与问题解决

        单例模式的初衷是为了控制对象的数量,使其不被无意创建。许多开发者面临对象管理不当导致的资源浪费和状态不一致的问题,而这一模式恰好能够有效解决这些问题。单例模式实现了对全局资源的集中管理,提高了系统的可维护性和安全性。

六、各语言实现与编码示例

Java 实现

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

Go 实现

package mainimport "sync"var instance *Singleton
var once sync.Oncetype Singleton struct{}func GetInstance() *Singleton {once.Do(func() {instance = &Singleton{}})return instance
}

C++ 实现

class Singleton {
private:static Singleton* instance;Singleton() {}
public:static Singleton* getInstance() {if (instance == nullptr) {instance = new Singleton();}return instance;}
};Singleton* Singleton::instance = nullptr;

Python 实现

class Singleton:_instance = Nonedef __new__(cls, *args, **kwargs):if not cls._instance:cls._instance = super(Singleton, cls).__new__(cls, *args, **kwargs)return cls._instancesingleton1 = Singleton()
singleton2 = Singleton()assert singleton1 is singleton2  # True

七、单例模式的优缺点

优点

  • 节省内存:同一时刻只存在一个实例。
  • 全局访问:提供全局访问点,方便在程序的不同部分共享资源。
  • 控制资源使用:通过限制实例化次数,可以避免资源浪费。

缺点

  • 可扩展性差:在某些场景下可能会导致系统中的“单点故障”。
  • 更难测试:单例模式可能使得测试变得复杂,因为它引入了全局状态。
  • 潜在的线程安全问题:在多线程环境中需谨慎处理,可能需要额外的同步机制。

八、单例模式的升级版

        在实际应用中,单例模式可能会演变成更复杂的形式,包括饿汉单例和懒汉单例。

饿汉单例

        饿汉单例在类被加载时就创建实例,因此实例的创建是提前的。通常使用一个静态常量来存储单例的实例。

特点

  • 线程安全:实例在类加载时就创建,因此不需要额外的同步机制。
  • 容易实现:比较简单,没有复杂的逻辑。
  • 资源浪费:即使没有使用,也会在加载时就创建实例,这可能导致资源浪费。

示例代码

public class HungrySingleton {// 在静态变量中创建唯一的实例private static final HungrySingleton INSTANCE = new HungrySingleton();// 私有构造函数,防止外部创建对象private HungrySingleton() {}// 提供一个全局访问点public static HungrySingleton getInstance() {return INSTANCE;}
}

懒汉单例

        懒汉单例在第一次被调用时才会创建实例,因此实例的创建是延迟的。通常会使用一个私有静态变量来存储实例,并在访问时检查是否已经被创建。

特点

  • 延迟加载:只有在需要的时候才创建实例,如果不需要,则不会创建。
  • 线程不安全:如果多个线程同时访问 getInstance() 方法,可能会导致创建多个实例。为了解决这个问题,常常在方法上添加同步机制。
  • 实现复杂:需要考虑多线程访问的安全性和性能。

示例代码(线程不安全):

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

示例代码(线程安全,使用同步):

public class ThreadSafeLazySingleton {private static ThreadSafeLazySingleton instance;private ThreadSafeLazySingleton() {}public static synchronized ThreadSafeLazySingleton getInstance() {if (instance == null) {instance = new ThreadSafeLazySingleton();}return instance;}
}
  • 饿汉单例适合于实例创建比较简单,并且应用程序总是需要这个实例的情况。
  • 懒汉单例适合于实例创建开销比较大,或者在特定条件下才需要创建的情况,但需要注意线程安全的问题。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/883626.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Spring Boot 实现文件上传下载功能

文章目录 一、原理分析1.1 请求类型1.2 服务器解析 二、功能实现2.1 创建项目并导入依赖2.2 文件上传功能实现2.2.1 文件上传 Service2.2.2 文件上传 Controller 2.3 文件下载功能实现2.3.1 文件下载 Service2.3.2 文件下载 Controller 2.4 文件上传前端代码(可选)2.4.1 上传文…

HarmonyOS Next API12最新版 端云一体化开发-云函数篇

一、新建一个端云一体化项目 见文章: HarmonyOS NEXT API12最新版 端云一体化开发-创建端云一体化项目流程_鸿蒙appapi-CSDN博客 二、官方文档 使用限制-云函数 - 华为HarmonyOS开发者 (huawei.com) Cloud Foundation Kit简介-Cloud Foundation Kit&#xff0…

深度学习:SGD的缺点

首先看下述函数: 最小值为x0,y0处 先了解下它的梯度特征。了理解其梯度特征,我们需要计算其梯度向量。 梯度向量 ∇f 是函数 f 在每个变量方向上的偏导数组成的向量。具体来说: ∇f(∂f/∂x,∂f∂/y) 首先,我们计算 f …

【北京迅为】《STM32MP157开发板嵌入式开发指南》-第六十七章 Trusted Firmware-A 移植

iTOP-STM32MP157开发板采用ST推出的双核cortex-A7单核cortex-M4异构处理器,既可用Linux、又可以用于STM32单片机开发。开发板采用核心板底板结构,主频650M、1G内存、8G存储,核心板采用工业级板对板连接器,高可靠,牢固耐…

Qt | windows视频播放器小项目

点击上方"蓝字"关注我们 01、前言 >>> Windows平台如果播放不了视频,记得下载编解码工具:https://www.mediaplayercodecpack.com/#google_vignette media.player.codec.pack.v4.6.0.setup.exe 下载后双击安装。 02、videowidget.pro >>> (.pro…

Android Activity SingleTop启动模式使用场景

通知栏 当用户点击通知栏中的通知时,可以使用单顶启动模式来打开对应的活动,并确保只有一个实例存在。 简单集成极光推送 创建应用 获取appkey参数 切换到极光工作台 极光sdk集成 Project 根目录的主 gradle 配置 Module 的 gradle 配置 Jpush依赖配置 配置推送必须…

ssm基于vue框架和elementui组件的手机官网+vue

系统包含:源码论文 所用技术:SpringBootVueSSMMybatisMysql 免费提供给大家参考或者学习,获取源码请私聊我 需要定制请私聊 目 录 目 录 III 1 绪论 1 1.1 研究背景 1 1.2 目的和意义 1 1.3 论文结构安排 2 2 相关技术 3 2.1 SSM框…

【Android】perfetto使用学习

在开发者选项中的系统跟踪里抓取的perfetto文件是保存在/data/local/traces 里的 adb pull /data/local/traces ./ 主线程中的执行是受vsync信号控制的,即间隔调用的 如果写一个while线程,一直使用cpu,是怎样的呢,这里我们来试验一…

技术周总结 10.21~10.27周日

文章目录 一、10.24 周四 程序员节2.1)问题01: Memory Analysis Tool的使用方法 二、10.27 周日2.1) J2EE架构J2EE 的核心组件与技术J2EE 的多层架构J2EE 的优缺点J2EE 的应用场景 2.2)web应用开发中的 "web服务器" 和 …

探讨Facebook的AI研究:未来社交平台的技术前瞻

在数字时代,社交媒体已成为人们日常生活的重要组成部分。作为全球最大的社交网络之一,Facebook不断致力于人工智能(AI)的研究与应用,以提升用户体验、增强平台功能并推动技术创新。本文将探讨Facebook在AI领域的研究方…

国标GB28181视频平台EasyGBS国标GB28181软件实现无需插件的视频监控对讲和网页直播

在当今社会,视频监控已经成为公共安全、企业管理、智能城市建设等领域不可或缺的一部分。然而,由于不同厂家和平台之间的兼容性问题,视频监控系统的联网和整合面临巨大挑战。为了解决这个问题,国家制定了《公共安全视频监控联网系…

【网络面试篇】三次握⼿、四次挥手综述

目录 一、三次握手 1. 过程描述 2. 为什么不是四次握手?为什么不能两次握手? 二、四次挥手 1. 过程描述 2. 为什么是四次挥手? 一、三次握手 1. 过程描述 ① 客户端 向 服务器 发送 SYN 报文、初始化序列号 ISN(seqx&…

JavaSet集合

无序(指的是添加顺序和获取出的数据顺序不一致,不重复,无索引 既然Set没有索引,因此功能同上一篇Connection的功能,几乎没有额外的功能 HashSet的原理 为什么是无序(要构建红黑树)&#xff0…

【测试平台】Odin-ws 工程环境部署

背景: 这个是我4年半以前接受测试平台过程中遇到问题记录,因为交接成都这边,拿出来直接用了。这里做个记录。 一、美东测试服务器相关 1.主服务器部署机器 该机器是美东服务器。 机器配置:t5.xlarge cpu 4核, 内存…

Android Junit 单元测试 | 依赖配置和编译报错解决

问题 为什么在依赖中添加了testImplement在build APK的时候还是会报错?是因为没有识别到test文件夹是test源代码路径吗? 最常见的配置有: implementation - 所有源代码集(包括test源代码集)中都有该依赖库.testImplementation - 依赖关系仅在test源代码…

如何将png格式的图片做成序列帧

类似以上的图片&#xff0c;根据图片的总长度和图片总数进行计算 <div class"frogeggitem"></div>.frogeggitem {width: 900rpx;height: 1000rpx;background: url(https://frog-skin.haoxgame.com/status2/frog/frogegg.png)no-repeat;background-size:…

学习笔记——动态路由——OSPF(距离矢量协议)OSPF路由类型

OSPF路由类型 在OSPF中&#xff0c;路由类型指的是不同种类的路由&#xff0c;用于描述网络中不同的路由信息及其传输方式。 1、Intra Area路由(区域内路由) Intra Area路由(区域内路由/本地路由/内部路由)是OSPF协议中的一种路由类型&#xff0c;用于描述在同一个OSPF区域内…

Android Studio Dolphin 下载、安装与配置教程

文章目录 Android Studio Dolphin简介一、核心特性二、新增功能三、用户体验优化 一&#xff0c;下载百度网盘迅雷云盘 二&#xff0c;安装三&#xff0c;下载组件四&#xff0c;添加SDK五&#xff0c;创建项目六&#xff0c;安装 Device模拟器运行项目 Android Studio Dolphin…

磁盘分区工具 DiskGenius Pro v5.5.0.1488 中文汉化版

DiskGenius 是一款专家级数据恢复软件&#xff0c;集数据恢复、硬盘分区、系统备份还原等多种功能于一身的超级工具软件&#xff0c;功能全面&#xff0c;安全可靠。可以提供磁盘的数据找回、备份、分区、修复、删除、格式化等操作&#xff0c;也能帮助用户修复磁盘坏道、彻底删…

《一人公司:失业潮中的高新技术工作者》读书笔记

基本信息 书名&#xff1a;一人公司&#xff1a;失业潮中的高新技术工作者作者&#xff1a;[美]卡丽莱恩&#xff08;Carrie M. Lane&#xff09;出版社&#xff1a;广东人民出版社出版时间&#xff1a;2023年11月 内容简介 本书通过18个月的田野调查和9年的跟踪寻访&#…