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


猜你喜欢
- 安装去http://www.mysql.com/downloads/, 选择最下方的MySQL Community Edition,点击My
- 1、说明Python实现异步IO非常简单,asyncio是Python 3.4版本引入的标准库,直接内置了对异步IO的支持。asyncio的
- 本文实例讲述了Python列表list操作符。分享给大家供大家参考,具体如下:#coding=utf8''''
- 本文实例为大家分享了python实现名片管理系统的具体代码,供大家参考,具体内容如下功能需求用户输入数字选择要进行的操作添加名片删除名片修改
- 最近做了一个项目其中有项目需求涉及到访问控制,在访问需要登录才能使用的页面或功能时,会弹出登录框:效果如下: 图 1-点击用户名时,如未登录
- 一.应用场景多个组件共用一个方法时可以用 mixin 抽取到一个js文件中,作为共用方法二.实现方法1.提取js共用方法文件export c
- 今天我升级MYSQL到5.1的时候遇到的。写出来共享以下。1、[root@localhost mysql]# scripts/mysql_i
- 安装好所需要的插件和包:python、django、pip等版本如下:采用Django REST框架3.01、在python文件夹下D:\p
- 一、基本用法1.注释Python中,#+语句 即为一条注释,也可以用 ‘'‘注释块 '''#人生苦短,我用P
- 引入Gridgrid=Grid() # 可以分别调整上下左右的位置,可以是百分比,也可以是具体像素,如pos_top="50px&
- 以下保存成 App.xml , 与asp文件放在相同目录下! 代码如下: <?xml version="1.0"
- 晚上打开MAC,发现root帐户突然不能正常登陆MySQL,于是打算重置密码,看了几篇文章,竟然重置不成功,总是得到Unknown colu
- <? //以树型结构列出指定目录里的所有文件,如果你想知道自己某个目录里有哪些子目录和文件,可以调用这个类来查看,很方便的。 &nbs
- sql_mode="",即强制不设定MySql模式(如不作输入检测、错误提示、语法模式检查等)应该能提高性能,但有如下问
- 1. 用户必须在几秒钟知道网站是做什么的。注意力是因特网上最有价值的货币。 如果访问者无法在几秒钟之内得知你的网站的方向,他很有可能转而访问
- 如何实现动态单行刷新,答案是——覆盖但是怎么实现覆盖呢关键在于不换行而且能回退到开始位置那么就要用到 \r这个东西就是让光标回退到
- 统一捕获接口报错弹窗提示报错重定向基础鉴权表单序列化实现的功能统一捕获接口报错 : 用的axios内置的 * 弹窗提示: 引入 E
- <%dim ylj,ywj,Mlpath,Shell,rarcomm,RetCode,cmd,comm,fsoM
- ORM框架使用最广泛的就是SQLAlchemy和Django自带的ORM框架,但是SQLAlchemy的语法显然相对Django的ORM框架
- python中的email模块可以方便的解析邮件,先上代码#-*- encoding: gb2312 -*-import osimport