Python实现单例模式

一、介绍

单例模式是一种常见的设计模式,它保证一个类只能被实例化一次,并提供了一个全局访问点来获取这个唯一的实例。在Python中,可以通过使用装饰器、元类或模块等方式实现单例模式。

二、Python实现单例模式的6种方法

1、使用模块实现单例

class Singleton(object):def foo(self):pass
singleton = Singleton()
from mysingleton import singleton
a = singleton
b = singleton
print(id(a))
print(id(b))

  

2、通过装饰器实现单例

def singleeton_func(cls):instance={}def _singleton(*args, **kwargs):if cls not in instance:instance[cls] = cls(*args, **kwargs)return instance[cls]return _singleton@singleeton_func
class Phone(object):def phone_id(self):return id(self)if __name__ == '__main__':p1 = Phone()p2 = Phone()print(p1.phone_id())print(p2.phone_id())

在装饰器的内函数中,判断字典是否已经有键值对。如果没有,则添加一个类和类实例的键值对,如果有,则不添加。最后返回字典中的类实例,可以保证每次返回的都是同一个实例。要使用这个单例装饰器,只要将其装饰到需要实现单例的类上即可。

3、使用实例化方式实现单例

class SingletonInstance(object):def __call__(self, *args, **kwargs):return self
if __name__ == '__main__':SingletonInstance = SingletonInstance()s1 = SingletonInstance()s2 = SingletonInstance()print(id(s1))print(id(s2))

 在类中,先重写类的 __call__ 方法,在 __call__ 方法中返回自己。先实例化一个类的对象,后面所有需要使用这个类的地方,都调用这个实例对象。这样,每次调用的都是同一个实例,所以也能实现单例。

4、使用类装饰器实现单例

class SingletonDecorator(object):_instance = Nonedef __init__(self, cls):self._cls = clsdef __call__(self, *args, **kwargs):if self._instance is None:self._instance = self._cls(*args, **kwargs)return self._instance@SingletonDecorator
class Phone(object):def phone_id(self):return id(self)if __name__ == '__main__':p1 = Phone()p2 = Phone()print(p1.phone_id())print(p2.phone_id())

_var:命名约定,仅供内部使用。通常不会由Python解释器强制执行。(私有变量)
var_:按约定使用以避免与Python关键字的命名冲突。
__var:当在类上下文中使用时,触发”名称修饰“。由Python解释器强制执行。
__var__:表示Python语言定义的特殊方法。避免在你自己的属性中使用这种命名方案。

使用装饰器实现单例,因为装饰器除了可以使用闭包实现,还可以使用类实现,所以也可以使用类装饰器来实现单例。通过重写类的 __call__ 方法实现类装饰器,在 __call__ 方法中判断当前是否有类实例,没有才会创建,从而实现单例。

5、重写类的__new__方法实现单例

python中的super()详解参考python中的super调用父类方法

class SingletonClass(object):_instance = Nonedef __new__(cls, *args, **kwargs):if cls._instance is None:# cls._instance = super(SingletonClass, cls).__new__(cls)  # python2.x中,super()函数的使用语法格式cls._instance = super().__new__(cls)  # python3.x中,super()函数的使用语法格式return cls._instance_is_init = Falsedef __init__(self):if self._is_init is False:print('-*-')self._is_init = Trueif __name__ == '__main__':s1 = SingletonClass()s2 = SingletonClass()print(id(s1))print(id(s2))

  

 在Python类中,每次实例化一个类对象时,都会自动先执行__new__方法和__init__方法。

__new__方法先在内存中为实例对象申请空间,然后__init__方法初始化实例对象。因为__init__方法是在new执行完成后自动执行的,每次实例类的对象时都会执行__init__方法,所以也要对__init__方法进行重写,只有第一次实例化类对象时才执行初始化操作。

通过重写__new__方法,如果这个类没有实例对象,则执行__new__,有则返回已有的实例,从而实现单例。

6、通过元类(metaclass)实现单例

元类详解参考Python中的元类

class SingletonMeta(type, object):def __init__(self, *args, **kwargs):self._instance = Nonesuper().__init__(*args, **kwargs)def __call__(self, *args, **kwargs):if self._instance is None:self._instance = super().__call__(*args, **kwargs)return self._instanceclass Phone(object, metaclass=SingletonMeta):def phone_id(self):return id(self)if __name__ == '__main__':p1 = Phone()p2 = Phone()print(p1.phone_id())print(p2.phone_id())

上述代码中,我们定义了一个名为SingletonMeta的元类。在元类的__call__方法中,我们首先判断该类是否已经存在于instances字典中,如果不存在,则创建一个新的实例并将其添加到instances字典中,否则返回已有的实例。 

 在python中,元类是创建类的类,是创建类的工厂,所有类的元类都是type类,所有类都是type类的实例对象。
 如果自定义一个元类,在元类中重写__call__方法,判断当前是否有实例,没有实例才创建,有则不创建。对需要实现单例的类,指定类的元类是我们自定义的元类,从而实现单例。(不推荐使用此方法)

三、单例模式应用场景

单例模式适用于需要确保一个类只有一个实例对象,并且该对象需要被全局访问的情况。
(1)资源共享的情况下,避免由于资源操作时导致的性能或损耗等。如日志文件、应用配置。
(2)控制资源的情况下,方便资源之间的互相通信。如线程池等。

实例--单例模式控制数据库连接池对象:

 假设我们正在开发一个多线程的应用程序,其中包含一个数据库连接池对象。为了避免在多个地方重复创建数据库连接池对象,我们可以使用单例模式来确保该对象只会被创建一次,并且在多个线程之间共享一个对象。

import threading
class DatabaseConnectionPool:instance = Nonelock = threading.Lock()def __new__(cls, *args, **kwargs):if cls.instance is None:with cls.lock:if cls.instance is None:cls.instance = super().__new__(cls)return cls.instancedef __init__(self):self.connections = []def add_connection(self, connection):self.connections.append(connection)def get_connection(self):return self.connections# 使用示例
pool = DatabaseConnectionPool()
pool.add_connection("connection1")
pool.add_connection("connection2")def worker(i):pool = DatabaseConnectionPool() # 多个线程共享同一个对象pool.add_connection("connection"+str(i+1))connection = pool.get_connection()print(f'Thread-{threading.get_ident()} got connection:{connection}')if __name__ == '__main__':for i in range(4):t = threading.Thread(target=worker(i))t.start()

上述代码中,我们首先定义了一个名为 DatabaseConnectionPool 的单例类,它维护了一个连接池列表 connections,通过 add_connection 和 get_connection 方法来添加和获取连接。使用 __new__ 方法来创建单例对象,确保在多个线程之间只有一个实例,同时使用锁来保证线程安全。
然后,我们在多个线程中使用同一个连接池对象,并通过 get_connection 方法来获取连接。由于所有的线程都共享同一个连接池对象,因此在获取连接时不会出现资源浪费和重复创建对象等问题。

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

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

相关文章

微软、OpenAI用上“数据永动机” 合成数据是晨曦还是暮光?

微软、OpenAI、Cohere等公司已经开始测试使用合成数据来训练AI模型。Cohere首席执行官Aiden Gomez表示,合成数据可以适用于很多训练场景,只是目前尚未全面推广。 已有的(通用)数据资源似乎接近效能极限,开发人员认为&a…

Java类的默认构造函数

什么情况下存在默认构造函数 说明 如果一个Java类没有显式包含构造函数的声明,那么隐含着有一个默认构造函数。 示例 定义一个类B,没有显式声明构造函数,所以存在一个默认构造函数: package com.thb;public class B {public …

selenium浏览器驱动下载

Chrome谷歌浏览器 下载地址:http://chromedriver.storage.googleapis.com/index.html 不同的Chrome的版本对应的chromedriver.exe 版本也不一样,下载时不要搞错了。 如果是最新的Chrome, 下载最新的chromedriver.exe 就可以了。 Firefox火狐浏览器 驱…

扫地机语音提示芯片,智能家居语音交互首选方案,WT588F02B-8S

智能家居已经成为现代家庭不可或缺的一部分,而语音交互技术正是智能家居的核心。在智能家居设备中,扫地机无疑是最受欢迎的产品之一。然而,要实现一个更智能的扫地机,需要一颗语音提示芯片,以提供高质量的语音交互体验…

Android Studio 的版本控制Git

Android Studio 的版本控制Git。 Git 是最流行的版本控制工具,本文介绍其在安卓开发环境Android Studio下的使用。 本文参考链接是:https://learntodroid.com/how-to-use-git-and-github-in-android-studio/ 一:Android Studio 中设置Git …

Linux系统安装部署Jenkins详细教程(图文讲解)

前言:最近需要使用Jenkins部署项目,所以想出一篇关于如何使用Linux系统安装部署Jenkins的相关教程,整体部署过程还是挺顺利的,特此分享一下! 目录 一、安装JDK11和Tomcat11 二、准备Jenkins安装包 三、部署Jenkins…

安全杂记 - js中的this关键字

javascript里什么是this this是js中的一个关键字&#xff0c;它是函数在运行时生成的一个内部对象&#xff0c;是属性和方法。 this就是属性或方法“当前”所在的对象&#xff0c;也就是调用函数的那个对象 this的使用场合 1.函数调用 <script>var a100;function test…

访问:http://localhost:8070/actuator/bus-refresh 问题

1、请求发送不出去 原因&#xff1a; 自己 config-server端 application.yml 配置的端口号是8888&#xff0c;访问server修改为配置的端口号 2、请求报错405 几个解决办法&#xff1a; 1、版本问题变为busrefresh 2、bus-refresh加单引号或双引号尝试 3、加配置尝试&#xff1a…

TypeScript -- 基础类型

文章目录 TypeScript -- 基础类型let 和 const基本类型写法布尔类型 -- boolean数字类型 -- number字符串类型 -- string数组类型元组类型枚举类型 -- enum任意类型 -- any空值 -- voidNull 和 Undefined不存在的类型 -- never对象 -- object类型断言 TypeScript – 基础类型 1…

【Linux下6818开发板(ARM)】SecureCRT串口和交叉编译工具(巨细版!)

(꒪ꇴ꒪ ),hello我是祐言博客主页&#xff1a;C语言基础,Linux基础,软件配置领域博主&#x1f30d;快上&#x1f698;&#xff0c;一起学习&#xff01;送给读者的一句鸡汤&#x1f914;&#xff1a;集中起来的意志可以击穿顽石!作者水平很有限&#xff0c;如果发现错误&#x…

【网络编程】(TCP流套接字编程 ServerSocket API Socket API 手写TCP版本的回显服务器 TCP中的长短连接)

文章目录 网络编程TCP流套接字编程ServerSocket APISocket APITCP中的长短连接手写TCP版本的回显服务器 网络编程 TCP流套接字编程 TCP提供的API主要是两个类:ServerSocket 和 Socket . TCP不需要一个类来表示"TCP数据报"因为TCP不是以数据报为单位进行传输的.是以…

管理类联考——数学——趣味篇——可视化

Manim: 一个数学可视化的动画引擎 官网&#xff1a;https://3b1b.github.io/manim/index.html 名词解析 python3.7是python语言的解释器, 运行python程序的环境必备品. 这个没啥说的,大家都能懂. 虽然官方建议3.7,但是我用3.8发现也没问题.考虑未来的历史进程,大伙最好还是装…

html中使用Vue+element UI动态创建表单数据不显示问题

直接上代码&#xff1a;html代码如下 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta http-equiv"X-UA-Compatible" content"IEedge"><meta name"viewport" content&…

Ansible的应用

Ansible简介 Ansible是一个基于Python开发的配置管理和应用部署工具&#xff0c;现在也在自动化管理领域大放异彩。它融合了众多老牌运维工具的优点&#xff0c;Pubbet和Saltstack能实现的功能&#xff0c;Ansible基本上都可以实现。 Ansible能批量配置、部署、管理上千台主机…

Io进、线程——进程的基础

进程的基础 进程是计算机中最基本的执行单位&#xff0c;是程序在操作系统中的一次执行过程。每个进程都有自己的地址空间、数据栈、程序计数器等&#xff0c;相互之间独立运行&#xff0c;互不干扰。进程间的通信通过特定的机制来实现&#xff0c;进程的创建和撤销由操作系统…

[AWD靶场搭建]

文章目录 [AWD靶场搭建]前言AWD平台搭建靶机搭建Cadinal添加靶机 连接Asteroid大屏默认ssh账号密码参考 [AWD靶场搭建] 前言 觉得好玩搭建了一下AWD靶场&#xff0c;使用了vidar-team编写的 Cardinal AWD平台搭建 这里我是在kali搭建的&#xff0c;所以我下载了这个压缩包&…

FANUC机器人SRVO-050碰撞检测报警和SRVO-053干扰值过大故障报警总结

FANUC机器人SRVO-050碰撞检测报警和SRVO-053干扰值过大故障报警总结 前面和大家分享了关于SRVO-050碰撞检测报警和SRVO-053干扰值过大的原因分析以及处理方法,感兴趣的朋友可以参考以下链接中的内容: FANUC机器人SRVO-050碰撞检测报警原因分析及处理对策

Java阶段五Day15

Java阶段五Day15 文章目录 Java阶段五Day15分层其他依赖dao-apidao-implinfrustructuredomainadaptermain 测试整合项目main前台师傅功能luban-front配置师傅相关表格ER图ER练习案例鲁班表格ER关系&#xff08;非常重要&#xff09; 前台师傅接口——师傅入驻adapterdomaininfr…

如何判断某个视频是深度伪造的?

目录 一、前言 二、仔细检查面部动作 三、声音可以提供线索 四、观察视频中人物的身体姿势 五、小心无意义的词语 深造伪造危险吗&#xff1f; 一、前言 制作深度伪造视频就像在Word文档中编辑文本一样简单。换句话说&#xff0c;您可以拍下任何人的视频&#xff0c;让他…

谷粒商城第六天-实现功能的前序工作(网关的配置 跨域配置)

目录 一、为什么要做这项工作 1.1 为什么要配置网关 1.2 为什么要使用网关统一配置跨域 二、网关配置 三、统一跨域配置 四、总结 一、为什么要做这项工作 1.1 为什么要配置网关 我们知道网关的作用其实主要就是进行路由的&#xff0c;也就是根据前端发送到网关的请求&…