设计模式-创建型-单例模式

1. 单例模式简介

单例模式(Singleton Pattern)是一种常见的创建型设计模式,它确保一个类只有一个实例,并提供全局访问点。在很多情况下,我们只希望某个类在整个应用程序中有一个唯一的实例,且该实例需要在整个系统中共享

单例模式的核心思想是:

  • 保证类只有一个实例:不管创建多少次,始终只有一个对象实例。
  • 提供全局访问点:通过某种方式让全局各处都能访问这个唯一的实例。

1.1 单例模式的应用场景

  • 资源共享:当系统中某些资源(如数据库连接、文件操作、线程池等)需要共享时,使用单例模式来保证它们只有一个实例。
  • 全局配置管理:例如在一个应用中有一个配置管理类,用于读取和保存系统的配置信息,使用单例模式可以确保配置管理类的实例不会被多次创建。
  • 日志管理:单例模式适用于日志管理器,在程序中仅有一个日志记录器实例,可以确保日志文件的统一管理。

2. 单例模式的实现

在 Python 中,单例模式有多种实现方式,常见的实现方式包括:通过__new__方法创建装饰器实现模块导入

2.1 通过__new__方法创建

class Singleton:_instance = Nonedef __new__(cls):if cls._instance is None:cls._instance = super(Singleton, cls).__new__(cls)return cls._instance# 测试饿汉式单例模式
singleton1 = Singleton()
singleton2 = Singleton()print(singleton1 is singleton2)  # 输出:True
解释:
  • __new__方法是用于创建实例的,它会在每次实例化时被调用。
  • __new__方法中,我们首先检查_instance是否为空,如果为空则创建实例并赋值给_instance。这样,无论如何,只会创建一个实例。

2.2 装饰器实现

可以使用Python的装饰器模式来实现单例模式。通过装饰器来控制类的实例化过程,确保类的实例唯一性。只要使用该装饰器,可以将任意类变为单例模式。

def singleton(cls):instances = {}def get_instance(*args, **kwargs):if cls not in instances:instances[cls] = cls(*args, **kwargs)return instances[cls]return get_instance@singleton
class Singleton:pass# 测试装饰器单例模式
singleton1 = Singleton()
singleton2 = Singleton()print(singleton1 is singleton2)  # 输出:True
解释:
  • singleton装饰器定义了一个instances字典,用于缓存已创建的实例。
  • 每次调用Singleton时,get_instance函数会检查是否已经有该类的实例存在,如果存在则直接返回缓存的实例,否则创建新的实例。

2.3 模块级单例

Python中的模块本身就是天然的单例。每个模块在首次导入时会被实例化,之后的导入会直接使用缓存的模块实例。这也是python常用的一种创建单例的方式。

# singleton_module.py
class Singleton:def __init__(self):print("Singleton instance created.")singleton_instance = Singleton() # singleton_module.py中先实例化,在其他模块中导入该实例# main.py
import singleton_module# 测试模块级单例模式
singleton1 = singleton_module.singleton_instance
singleton2 = singleton_module.singleton_instanceprint(singleton1 is singleton2)  # 输出:True
解释:
  • 模块本身在导入时会被加载并执行一次,这样singleton_instance就会成为一个单例实例,后续的导入会直接引用这个实例。

3. 单例模式的优缺点

3.1 优点

  • 节约内存:单例模式确保系统中只有一个实例,避免了多次创建对象带来的内存浪费。
  • 全局访问:通过单例模式提供了一个全局访问点,保证了全局共享的数据。

3.2 缺点

  • 不易于测试:单例模式难以模拟和控制单例对象,可能影响单元测试。
  • 潜在的并发问题:在多线程环境下,单例模式可能导致并发问题,需要额外的小心,避免在多线程环境中创建多个实例。
  • 违反单一职责原则:单例模式将实例的管理逻辑和业务逻辑放在同一个类中,可能导致类的职责过于庞大。

4. 结语

单例模式是一个非常常见的设计模式,广泛应用于系统中需要共享资源、配置管理、日志等场景。它通过控制实例的创建过程确保了类只有一个实例,且提供了全局访问点。理解单例模式并合理使用它可以提升代码的可维护性和灵活性。

尽管单例模式有很多优点,但也需要谨慎使用,特别是在多线程环境下,确保不会导致并发问题。此外,单例模式可能会使得代码耦合度过高,影响代码的扩展性和测试性,因此在使用时要根据实际情况权衡其利弊。

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

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

相关文章

全面Kafka监控方案:从配置到指标

文章目录 1.1.监控配置1.2.监控工具1.3.性能指标系统相关指标GC相关指标JVM相关指标Topic相关指标Broker相关指标 1.4.性能指标说明1.5.重要指标说明 1.1.监控配置 开启JMX服务端口:kafka基本分为broker、producer、consumer三个子项,每一项的启动都需要…

网络安全威胁2024年中报告

下载地址: 网络安全威胁2024年中报告-奇安信

AI辅助编码提高病案首页主要诊断编码正确率数据优化方法(2025增量优化版附python源代码)

一、引言 1.1 研究背景与意义 在医疗信息化进程中,病案首页作为病历信息的核心浓缩,承载着疾病分类、医疗统计、医保结算等关键任务,其主要诊断编码的准确性至关重要。准确的编码不仅是医疗质量评估、科研数据分析的基石,更是合理分配医疗资源、保障医保精准支付的关键依…

雷电模拟器安装LSPosed

雷电模拟器最新版支持LSPosed。记录一下安装过程 首先到官网下载并安装最新版,我安装的时候最新版是9.1.34.0,64位 然后开启root和系统文件读写 然后下载magisk-delta-6并安装 ,这个是吾爱破解论坛提供的,号称适配安卓7以上所有机型&#x…

模型 10-10-10旁观思维

系列文章 分享 模型,了解更多👉 模型_思维模型目录。超脱当下,透视决策长远影响。 1 10-10-10旁观思维的应用 1.1 职业选择决策 背景:小张是一名大学毕业生,面对未来职业的选择感到迷茫。他擅长营销、策略和经济学&a…

ORM框架详解:为什么不直接写SQL?

想象一下,你正在开发一个小型的在线书店应用。你需要存储书籍信息、用户数据和订单记录。作为一个初学者,你可能会想:“我已经学会了SQL,为什么还要使用ORM框架呢?直接写SQL语句不是更简单、更直接吗?” 如…

合合信息亮相CSIG AI可信论坛,全面拆解AI视觉内容安全的“终极防线”

合合信息亮相CSIG AI可信论坛,全面拆解视觉内容安全的“终极防线”! 🐯 AI伪造泛滥,我们还能相信“眼见为实”吗? 近期,由中国图象图形学学会主办的CSIG青年科学家会议 AI可信论坛在杭州成功举办。本次论…

Linux实验报告9-进程管理

目录 一:实验目的 二:实验内容 (1)列出当前系统中的所有进程,如何观察进程的优先级? (2)查看当前终端运行的 bash 进程的 PID,在当前终端启动 vim 编辑器并让其在后台执行,然后列出在当前终端中执行的进程的家族树。 (3)请自行挂载U盘或光盘,然后…

17爬虫:关于DrissionPage相关内容的学习01

概述 前面我们已经大致了解了selenium的用法,DerssionPage同selenium一样,也是一个基于Python的网页自动化工具。 DrissionPage既可以实现网页的自动化操作,也能够实现收发数据包,也可以把两者的功能合二为一。 DressionPage的…

【Unity3D】Jobs、Burst并行计算裁剪Texture3D物体

版本:Unity2019.4.0f1 PackageManager下载Burst插件(1.2.3版本) 利用如下代码,生成一个Texture3D资源,它只能脚本生成,是一个32*32*32的立方体,导出路径记得改下,不然报错。 using UnityEditor; using Uni…

最短路径-Dijkstra 算法

前言 迪杰斯特拉算法是由荷兰计算机科学家狄克斯特拉于1959 年提出的,因此又叫狄克斯特拉算法。是从一个顶点到其余各顶点的最短路径算法,解决的是有向图中最短路径问题。迪杰斯特拉算法主要特点是以起始点为中心向外层层扩展,直到扩展到终点…

ESP32 I2S音频总线学习笔记(一):初识I2S通信与配置基础

文章目录 简介为什么需要I2S?关于音频信号采样率分辨率音频声道 怎样使用I2S传输音频?位时钟BCLK字时钟WS串行数据SD I2S传输模型I2S通信格式I2S格式左对齐格式右对齐格式 i2s基本配置i2s 底层API加载I2S驱动设置I2S使用的引脚I2S读取数据I2S发送数据卸载…

Eclipse中引入NS3项目

参考资料: 博主:深度不睡觉 NS3的3.36版本将Eclipse作IDE_ns3使用eclipse-CSDN博客 从1.2安装eclipse开始 其中参考教程中省略的几点: 1.下载解压tar包 mkdir /Tools/Eclipse/EclipseTool # 新建目录 tar -zxvf /path/to/eclipse-cpp-20…

机器学习周报-TCN文献阅读

文章目录 摘要Abstract 1 TCN通用架构1.1 序列建模任务描述1.2 因果卷积(Causal Convolutions)1.3 扩张卷积(Dilated Convolutions)1.4 残差连接(Residual Connections) 2 TCN vs RNN3 TCN缺点4 代码4.1 TC…

Quartz任务调度框架实现任务动态执行

说明:之前使用Quartz,都是写好Job,指定一个时间点,到点执行。最近有个需求,需要根据前端用户设置的时间点去执行,也就是说任务执行的时间点是动态变化的。本文介绍如何用Quartz任务调度框架实现任务动态执行…

Spring-kafka快速Demo示例

使用Spring-Kafka快速发送/接受Kafka消息示例代码&#xff0c;项目结构是最基础的SpringBoot结构&#xff0c;提前安装好Kafka&#xff0c;确保Kafka已经正确启动 pom.xml&#xff0c;根据个人情况更换springboot、java版本等 <?xml version"1.0" encoding&qu…

【C++】B2079 求出 e 的值

博客主页&#xff1a; [小ᶻ☡꙳ᵃⁱᵍᶜ꙳] 本文专栏: C 文章目录 &#x1f4af;前言&#x1f4af;题目介绍输入格式输出格式输入输出样例说明/提示 &#x1f4af;实现方法一&#xff1a;单层 for 循环计算代码实现运行逻辑解析优点不足 &#x1f4af;实现方法二&#xff…

STM32配合可编程加密芯片SMEC88ST的防抄板加密方案设计

SMEC88ST SDK卡发包下载 目前市场上很多嵌入式产品方案都是可以破解复制的&#xff0c;主要是因为方案主芯片不具备防破解的功能&#xff0c;这就导致开发者投入大量精力、财力开发的新产品一上市就被别人复制&#xff0c;到市场上的只能以价格竞争&#xff0c;最后工厂复制的产…

精准识别花生豆:基于EfficientNetB0的深度学习检测与分类项目

精准检测花生豆&#xff1a;基于EfficientNet的深度学习分类项目 在现代农业生产中&#xff0c;作物的质量检测和分类是确保产品质量的重要环节。针对花生豆的检测与分类需求&#xff0c;我们开发了一套基于深度学习的解决方案&#xff0c;利用EfficientNetB0模型实现高效、准…

MarkItDown的使用(将Word、Excel、PDF等转换为Markdown格式)

MarkItDown的使用&#xff08;将Word、Excel、PDF等转换为Markdown格式&#xff09; 本文目录&#xff1a; 零、时光宝盒&#x1f33b; 一、简介 二、安装 三、使用方法 3.1、使用命令行形式 3.2、用 Python 调用 四、总结 五、参考资料 零、时光宝盒&#x1f33b; &a…