python中单例常用的几种实现方法总结
作者:python修行路 发布时间:2022-02-28 19:37:55
标签:python,单例,常用
前言
最近这两天在看自己之前写的代码,所以正好把用过的东西整理一下,单例模式,在日常的代码工作中也是经常被用到,
所以这里把之前用过的不同方式实现的单例方式整理一下
什么是单例?
确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例,这个类称为单例类,单例模式是一种对象创建型模式。
那么单例模式有什么用途呢?举个常见的单例模式例子,我们平时使用的电脑上都有一个回收站,在整个操作系统中,回收站只能有一个实例,整个系统都使用这个唯一的实例,而且回收站自行提供自己的实例,因此回收站是单例模式的应用。
装饰器的方式
这种方式也是工作中经常用的一种,用起来也比较方便,代码实现如下
def Singleton(cls):
_instance = {}
def _singleton(*args, **kwargs):
if cls not in _instance:
_instance[cls] = cls(*args, **kwargs)
return _instance[cls]
return _singleton
如果我们工作的一个类需要用单例就通过类似下面的方式实现即可:
@Singleton
class A(object):
def __init__(self, x):
self.x = x
我个人还是挺喜欢这种方式的
类的方式实现
这里其实有一些问题就需要注意了,先看一下可能出现的错误代码
class Member(object):
@classmethod
def instance(cls, *args, **kwargs):
if not hasattr(Member, "_instance"):
Member._instance = Member(*args, **kwargs)
return Member._instance
乍一看这个类好像已经实现了单例,但是这里有一个潜在的问题,就是如果是多线程的情况,这样写就会有问题了,尤其是在当前类的初始化对象里有一些耗时操作时候
例如下面代码:
#! /usr/bin/env python3
# .-*- coding:utf-8 .-*-
import time
import threading
import random
class Member(object):
def __init__(self):
time.sleep(random.randint(1,3))
@classmethod
def instance(cls, *args, **kwargs):
if not hasattr(Member, "_instance"):
Member._instance = Member(*args, **kwargs)
return Member._instance
def task(arg):
obj = Member.instance()
print(obj)
for i in range(5):
t = threading.Thread(target=task, args=[i,])
t.start()
这段代码的执行结果会出现实例化了多个对象,导致你写的单例就没起到作用
当然自然而然我们会想起加锁,通过锁来控制,所以我们将上面代码进行更改:
#! /usr/bin/env python3
# .-*- coding:utf-8 .-*-
import time
import threading
import random
class Member(object):
_instance_lock = threading.Lock()
def __init__(self):
i = random.randint(1, 3)
print(i)
time.sleep(i)
@classmethod
def instance(cls, *args, **kwargs):
with Member._instance_lock:
if not hasattr(Member, "_instance"):
Member._instance = Member(*args, **kwargs)
return Member._instance
def task():
obj = Member.instance()
print(obj)
for i in range(5):
threading.Thread(target=task,).start()
但是上面的代码还有一个问题,就是当我们已经实例化过之后每次调用instance都会去请求锁,所以这点并不好,所以我们将这部分代码再次更改:
@classmethod
def instance(cls, *args, **kwargs):
if not hasattr(Member, "_instance"):
with Member._instance_lock:
if not hasattr(Member, "_instance"):
Member._instance = Member(*args, **kwargs)
return Member._instance
这样就很好的实现一个可以多线程使用的单例
来源:https://www.cnblogs.com/zhaof/p/9774525.html
0
投稿
猜你喜欢
- 引入依赖#?导入模块import?pymysqlimport?pandas?as?pdimport?numpy?as?npimport?ti
- 命令:Python manage.py runserver改为python manage.py runserver 0.0.0.0:80外网
- python编程中,我们在修改代码,遇到空格很多的情况下,我们要删除空格。本文小编整理了三种字符串去除空格的方法:方法一:使用字符串函数re
- 1 前言正则表达式是对字符串(包括普通字符(例如,a 到 z 之间的字母)和特殊字符(称为“元字符”))操作的一种逻辑公式,就是用事先定义好
- 一.权限表mysql数据库中的3个权限表:user 、db、 host权限表的存取过程是:1)先从user表中的host、 user、 pa
- 完整的Oracle数据库通常由两部分组成:Oracle数据库和数据库实例。 1) 数据库是一系列物理文件的集合(数据文件,控制文件,联机日志
- 本文实例讲述了python实现的自动发送消息功能。分享给大家供大家参考,具体如下:一个简单的脚本#-*- coding:utf-8 -*-f
- asp使用WScript.Shell获取电脑的网络配置信息Option Explicit Dim WSHShe
- 本文实例讲述了Python实现输入二叉树的先序和中序遍历,再输出后序遍历操作。分享给大家供大家参考,具体如下:实现一个功能: &n
- 什么是索引?哪些数据类型里有索引的概念? —> 字符串、列表、元组从最左边记录的位置开始就是索引索引用数字表示,起
- 前言:我们在日常生活中,都离不开时间和日期。不仅我们的汉字都有大量描述时间日期的词语如斗转星移、分秒必争、只针朝夕、转眼之间等。同样,在我们
- php高并发之opcache今天工作的时候接触到客户的一台服务器,业务逻辑比较简单 。估算pv在120w左右吧,用的是阿里云2c4g的服务器
- 项目地址:https://github.com/Henryhaohao/Bilibili_video_download介绍对于单P视频:直接
- 前言:本文的主要内容是介绍Python中的变量命名规则和简单数据类型的应用,简单的数据类型包括字符串和数字等,文中还附有代码以及相应的运行结
- 音乐流媒体服务的兴起使得音乐无处不在。我们在上下班的时候听音乐,锻炼身体,工作或者只是放松一下。这些服务的一个关键特性是播放列表,通常按流派
- # -*- coding: utf-8 -*-# 测试各种排序算法# link:www.jb51.net# date:2013/2/2#选择
- 目录房价数据分析数据简单清洗各区均价分析全市二手房装修程度分析各区二手房数量所占比比例热门户型均价分析总结房价数据分析数据简单清洗data.
- 0.引言利用python开发,借助Dlib库捕获摄像头中的人脸,提取人脸特征,通过计算欧氏距离来和预存的人脸特征进行对比,达到人脸识别的目的
- 我们也可以来做一个,但这个“定时器”的工作时间范围应控制在1个小时至100 毫秒之间: <%sub StartTi
- 前言argsparse是python的命令行解析的标准模块,内置于python,不需要安装。这个库可以让我们直接在命令行中就可以向程序中传入