举例说明python单利模式的必要性

单例模式的核心目的是确保一个类只有一个实例,并提供一个全局访问点来获取这个实例。这种设计模式在某些场景下非常必要,尤其是在需要严格控制资源访问、共享状态或配置管理的场景中。下面通过几个具体的例子来说明Python中单例模式的必要性。


1. 数据库连接池

在应用程序中,频繁地创建和销毁数据库连接是非常低效的。使用单例模式可以确保整个应用程序共享一个数据库连接池,从而减少资源开销。

示例:
class DatabaseConnectionPool:_instance = Nonedef __new__(cls):if cls._instance is None:cls._instance = super().__new__(cls)cls._instance.init_pool()return cls._instancedef init_pool(self):# 模拟初始化连接池self.pool = ["Connection1", "Connection2", "Connection3"]print("Database connection pool initialized.")def get_connection(self):if self.pool:return self.pool.pop()else:raise Exception("No available connections.")def release_connection(self, connection):self.pool.append(connection)print(f"Connection {connection} released.")# 使用单例模式
pool1 = DatabaseConnectionPool()
pool2 = DatabaseConnectionPool()print(pool1 is pool2)  # 输出: Trueconn1 = pool1.get_connection()
print(conn1)  # 输出: Connection1pool1.release_connection(conn1)  # 输出: Connection Connection1 released.
必要性:
  • 避免重复创建连接池,节省资源。
  • 确保所有模块共享同一个连接池,避免连接泄露或资源竞争。

2. 配置管理

在应用程序中,通常需要一个全局的配置管理器来读取和存储配置信息。使用单例模式可以确保配置信息在整个应用程序中保持一致。

示例:
class ConfigManager:_instance = Nonedef __new__(cls):if cls._instance is None:cls._instance = super().__new__(cls)cls._instance.load_config()return cls._instancedef load_config(self):# 模拟加载配置文件self.config = {"host": "localhost","port": 8080,"debug": True}print("Configuration loaded.")def get_config(self, key):return self.config.get(key)# 使用单例模式
config1 = ConfigManager()
config2 = ConfigManager()print(config1 is config2)  # 输出: Trueprint(config1.get_config("host"))  # 输出: localhost
print(config2.get_config("port"))  # 输出: 8080
必要性:
  • 避免重复加载配置文件,节省时间和内存。
  • 确保所有模块访问的是同一份配置数据,避免配置不一致。

3. 日志记录器

在应用程序中,通常需要一个全局的日志记录器来统一管理日志输出。使用单例模式可以确保所有模块使用同一个日志记录器。

示例:
import loggingclass Logger:_instance = Nonedef __new__(cls):if cls._instance is None:cls._instance = super().__new__(cls)cls._instance.init_logger()return cls._instancedef init_logger(self):self.logger = logging.getLogger("AppLogger")self.logger.setLevel(logging.INFO)handler = logging.StreamHandler()formatter = logging.Formatter('%(asctime)s - %(name)s - %(levelname)s - %(message)s')handler.setFormatter(formatter)self.logger.addHandler(handler)print("Logger initialized.")def log(self, message):self.logger.info(message)# 使用单例模式
logger1 = Logger()
logger2 = Logger()print(logger1 is logger2)  # 输出: Truelogger1.log("This is a log message.")  # 输出: 2023-10-01 12:00:00 - AppLogger - INFO - This is a log message.
必要性:
  • 避免重复创建日志记录器,节省资源。
  • 确保所有模块使用同一个日志记录器,避免日志输出混乱。

4. 缓存系统

在应用程序中,通常需要一个全局的缓存系统来存储临时数据。使用单例模式可以确保所有模块共享同一个缓存实例。

示例:
class Cache:_instance = Nonedef __new__(cls):if cls._instance is None:cls._instance = super().__new__(cls)cls._instance.init_cache()return cls._instancedef init_cache(self):self.cache = {}print("Cache initialized.")def set(self, key, value):self.cache[key] = valuedef get(self, key):return self.cache.get(key)# 使用单例模式
cache1 = Cache()
cache2 = Cache()print(cache1 is cache2)  # 输出: Truecache1.set("user1", "Alice")
print(cache2.get("user1"))  # 输出: Alice
必要性:
  • 避免重复创建缓存实例,节省内存。
  • 确保所有模块访问的是同一份缓存数据,避免数据不一致。

5. 硬件设备控制

在嵌入式系统或硬件控制应用中,通常需要确保只有一个实例来控制硬件设备(如打印机、传感器等)。使用单例模式可以避免多个实例同时操作硬件导致的冲突。

示例:
class PrinterController:_instance = Nonedef __new__(cls):if cls._instance is None:cls._instance = super().__new__(cls)cls._instance.init_printer()return cls._instancedef init_printer(self):# 模拟初始化打印机self.status = "Ready"print("Printer initialized.")def print_document(self, document):if self.status == "Ready":print(f"Printing: {document}")else:print("Printer is busy.")# 使用单例模式
printer1 = PrinterController()
printer2 = PrinterController()print(printer1 is printer2)  # 输出: Trueprinter1.print_document("Document1")  # 输出: Printing: Document1
printer2.print_document("Document2")  # 输出: Printer is busy.
必要性:
  • 避免多个实例同时操作硬件设备,防止冲突或损坏。
  • 确保硬件资源被合理管理和使用。

总结

单例模式的必要性主要体现在以下几个方面:

  1. 资源管理:避免重复创建和销毁资源(如数据库连接、缓存、日志记录器等)。
  2. 状态共享:确保全局状态的一致性(如配置管理、硬件控制)。
  3. 性能优化:减少内存和计算资源的浪费。
  4. 冲突避免:防止多个实例同时操作共享资源导致的冲突。

在实际开发中,单例模式应谨慎使用,避免过度设计。只有在确实需要全局唯一实例的场景下,才推荐使用单例模式。

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

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

相关文章

【腾讯云】腾讯云docker搭建单机hadoop

这里写目录标题 下载jdk hadoop修改hadoop配置编写Dockerfile构建镜像运行镜像创建客户端 下载jdk hadoop wget --no-check-certificate https://repo.huaweicloud.com/java/jdk/8u151-b12/jdk-8u151-linux-x64.tar.gz wget --no-check-certificate https://repo.huaweicloud.…

设计模式 - 行为模式_Template Method Pattern模板方法模式在数据处理中的应用

文章目录 概述1. 核心思想2. 结构3. 示例代码4. 优点5. 缺点6. 适用场景7. 案例:模板方法模式在数据处理中的应用案例背景UML搭建抽象基类 - 数据处理的 “总指挥”子类定制 - 适配不同供应商供应商 A 的数据处理器供应商 B 的数据处理器 在业务代码中整合运用 8. 总…

HTML5+SVG+CSS3实现雪中点亮的圣诞树动画效果源码

源码介绍 这是一款基于HTML5SVGCSS3实现雪中点亮的圣诞树动画效果源码。画面中的圣诞树矗立在雪地中,天上飘落着雪花。当鼠标滑过圣诞树时,可见到圣诞树上的灯光闪烁,同时左下角探出雪怪模样的半个脑袋,四处张望着。整体画面栩栩…

C基础寒假练习(3)

一、求数组中的第二大值 #include <stdio.h> int main() {int arr[] {12, 35, 1, 10, 34, 1};int size sizeof(arr) / sizeof(arr[0]);if (size < 2) {printf("数组元素不足两个\n");return 0;}int first -2147483648, second -2147483648; // 使用IN…

【linux三剑客】grep练习题

题目 进入/lianxi目录&#xff0c;复制/etc/passwd到当前目录下&#xff0c;然后对passwd进行操作查找出当前passwd文件中以ftp或者mail开头的行&#xff0c;在屏幕上输出。查找出当前passwd文件中有没有以r、m、f开头的行&#xff0c;在屏幕上输出。查找出当前passwd文件中以…

C++,STL,【目录篇】

文章目录 一、简介二、内容提纲第一部分&#xff1a;STL 概述第二部分&#xff1a;STL 容器第三部分&#xff1a;STL 迭代器第四部分&#xff1a;STL 算法第五部分&#xff1a;STL 函数对象第六部分&#xff1a;STL 高级主题第七部分&#xff1a;STL 实战应用 三、写作风格四、…

【Node.js】Koa2 整合接口文档

部分学习来源&#xff1a;https://blog.csdn.net/qq_38734862/article/details/107715579 依赖 // koa2-swagger-ui UI视图组件 swagger-jsdoc 识别写的 /***/ 转 json npm install koa2-swagger-ui swagger-jsdoc --save配置 config\swaggerConfig.js const Router requir…

Maven的单元测试

1. 单元测试的基本概念 单元测试&#xff08;Unit Testing&#xff09; 是一种软件测试方法&#xff0c;专注于测试程序中的最小可测试单元——通常是单个类或方法。通过单元测试&#xff0c;可以确保每个模块按预期工作&#xff0c;从而提高代码的质量和可靠性。 2.安装和配…

论文阅读(八):结构方程模型用于研究数量遗传学中的因果表型网络

1.论文链接&#xff1a;Structural Equation Models for Studying Causal Phenotype Networks in Quantitative Genetics 摘要&#xff1a; 表型性状可能在它们之间发挥因果作用。例如&#xff0c;农业物种的高产可能会增加某些疾病的易感性&#xff0c;相反&#xff0c;疾病的…

LeetCode | 不同路径

一个机器人位于一个 m x n 网格的左上角 &#xff08;起始点在下图中标记为 “Start” &#xff09;。 机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角&#xff08;在下图中标记为 “Finish” &#xff09;。 问总共有多少条不同的路径&#xff1f; 示例 1…

C++的类Class

文章目录 一、C的struct和C的类的区别二、关于OOP三、举例&#xff1a;一个商品类CGoods四、构造函数和析构函数1、定义一个顺序栈2、用构造和析构代替s.init(5);和s.release();3、在不同内存区域构造对象4、深拷贝和浅拷贝5、构造函数和深拷贝的简单应用6、构造函数的初始化列…

Excel 技巧21 - Excel中整理美化数据实例,Ctrl+T 超级表格(★★★)

本文讲Excel中如何整理美化数据的实例&#xff0c;以及CtrlT 超级表格的常用功能。 目录 1&#xff0c;Excel中整理美化数据 1-1&#xff0c;设置间隔行颜色 1-2&#xff0c;给总销量列设置数据条 1-3&#xff0c;根据总销量设置排序 1-4&#xff0c;加一个销售趋势列 2&…

Leetcode 131 分割回文串(纯DFS)

131. 分割回文串https://leetcode.cn/problems/palindrome-partitioning/https://leetcode.cn/problems/palindrome-partitioning/ 给你一个字符串 s&#xff0c;请你将 s 分割成一些子串&#xff0c;使每个子串都是 回文串 。返回 s 所有可能的分割方案。 示例 1&#xff1a…

电梯系统的UML文档14

对于 HallButtonControl&#xff0c;我们有二个状态: "门厅灯开 " 和 " 门厅灯关"。 从给出的初始信息&#xff0c;初始的状态应该是"门厅灯关"。行为定义&#xff1a; " 当 HallCall[f&#xff0c;d]是真&#xff0c;则指令 HallLight[f&…

关于安卓greendao打包时报错问题修复

背景 项目在使用greendao的时候&#xff0c;debug安装没有问题&#xff0c;一到打包签名就报了。 环境 win10 jdk17 gradle8 项目依赖情况 博主的greendao是一个独立的module项目&#xff0c;项目目前只适配了java&#xff0c;不支持Kotlin。然后被外部集成。greendao版本…

SQL server 数据库使用整理

标题&#xff1a;SQL server 数据库使用整理 1.字符串表名多次查询 2.读取SQL中Json字段中的值&#xff1a;JSON_VALUE&#xff08;最新版本支持&#xff0c;属性名大小写敏感&#xff09; 1.字符串表名多次查询 SELECT ROW_NUMBER() OVER (ORDER BY value ASC) rowid,value…

一文讲解Java中的BIO、NIO、AIO之间的区别

BIO、NIO、AIO是Java中常见的三种IO模型 BIO&#xff1a;采用阻塞式I/O模型&#xff0c;线程在执行I/O操作时被阻塞&#xff0c;无法处理其他任务&#xff0c;适用于连接数比较少的场景&#xff1b;NIO&#xff1a;采用非阻塞 I/O 模型&#xff0c;线程在等待 I/O 时可执行其…

分布式系统架构怎么搭建?

分布式系统架构 互联网企业的业务飞速发展&#xff0c;促使系统架构不断变化。总体来说&#xff0c;系统架构大致经历了单体应用架构—垂直应用架构—分布式架构—SOA架构—微服务架构的演变&#xff0c;很多互联网企业的系统架构已经向服务化网格&#xff08;Service Mesh&am…

Effective C++ 规则50:了解 new 和 delete 的合理替换时机

1、背景 在 C 中&#xff0c;new 和 delete 是动态分配内存的核心操作符。然而&#xff0c;直接使用它们有时会增加程序的复杂性&#xff0c;甚至导致内存泄漏和其他问题。因此&#xff0c;了解何时替换 new 和 delete 并选择更适合的内存管理策略&#xff0c;是编写高效、健壮…

Effective Python:(10)

Effective Python提供90条新颖的Python3编程技巧&#xff0c;可以让我们写程序更加灵活&#xff0c;代码更加整洁而易于维护&#xff0c;这对于商业化系统代码的重要性不言而喻。 前面两条主要介绍切片的实用好玩的用法&#xff0c;这一条里反而建议不用切片&#xff0c;这是什…