python中单例常用的几种实现方法总结


Posted in Python onOctober 13, 2018

前言

最近这两天在看自己之前写的代码,所以正好把用过的东西整理一下,单例模式,在日常的代码工作中也是经常被用到,

所以这里把之前用过的不同方式实现的单例方式整理一下

什么是单例?

确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例,这个类称为单例类,单例模式是一种对象创建型模式。

那么单例模式有什么用途呢?举个常见的单例模式例子,我们平时使用的电脑上都有一个回收站,在整个操作系统中,回收站只能有一个实例,整个系统都使用这个唯一的实例,而且回收站自行提供自己的实例,因此回收站是单例模式的应用。

装饰器的方式

这种方式也是工作中经常用的一种,用起来也比较方便,代码实现如下

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

这样就很好的实现一个可以多线程使用的单例

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对三水点靠木的支持。

Python 相关文章推荐
python文本数据处理学习笔记详解
Jun 17 Python
详解Python中的各种转义符\n\r\t
Jul 10 Python
Django中reverse反转并且传递参数的方法
Aug 06 Python
pytorch numpy list类型之间的相互转换实例
Aug 18 Python
Python 实现Image和Ndarray互相转换
Feb 19 Python
详解python内置常用高阶函数(列出了5个常用的)
Feb 21 Python
Python callable内置函数原理解析
Mar 05 Python
Python数据结构dict常用操作代码实例
Mar 12 Python
python让函数不返回结果的方法
Jun 22 Python
pandas apply多线程实现代码
Aug 17 Python
Python和Bash结合在一起的方法
Nov 13 Python
python​格式化字符串
Apr 20 Python
python中pika模块问题的深入探究
Oct 13 #Python
Pycharm无法使用已经安装Selenium的解决方法
Oct 13 #Python
解决python selenium3启动不了firefox的问题
Oct 13 #Python
selenium + python 获取table数据的示例讲解
Oct 13 #Python
Python3.4 splinter(模拟填写表单)使用方法
Oct 13 #Python
小白入门篇使用Python搭建点击率预估模型
Oct 12 #Python
Python读取txt内容写入xls格式excel中的方法
Oct 11 #Python
You might like
dedecms防止FCK乱格式化你的代码的修改方法
2007/03/17 PHP
Gregarius中文日期格式问题解决办法
2008/04/22 PHP
分享一个PHP数据流应用的简单例子
2012/06/01 PHP
php对数组排序的简单实例
2013/12/25 PHP
PHP写日志的实现方法
2014/11/05 PHP
Ubuntu下安装PHP的mongodb扩展操作命令
2015/07/04 PHP
从jQuery.camelCase()学习string.replace() 函数学习
2011/09/13 Javascript
解析jQuery的三种bind/One/Live事件绑定使用方法
2013/12/30 Javascript
jquery移除、绑定、触发元素事件使用示例详解
2014/04/10 Javascript
用window.onerror捕获并上报Js错误的方法
2016/01/27 Javascript
js判断图片加载完成后获取图片实际宽高的方法
2016/02/25 Javascript
JS使用面向对象技术实现的tab选项卡效果示例
2017/02/28 Javascript
bootstrap datepicker插件默认英文修改为中文
2017/07/28 Javascript
ios中视频的最后一桢问题解决
2019/05/14 Javascript
js实现图片推拉门效果代码实例
2019/05/18 Javascript
vue Treeselect下拉树只能选择第N级元素实现代码
2020/08/31 Javascript
Python中用函数作为返回值和实现闭包的教程
2015/04/27 Python
在Ubuntu系统下安装使用Python的GUI工具wxPython
2016/02/18 Python
python遍历 truple list dictionary的几种方法总结
2016/09/11 Python
Python常见字典内建函数用法示例
2018/05/14 Python
Python中变量的输入输出实例代码详解
2019/07/28 Python
Django获取应用下的所有models的例子
2019/08/30 Python
解决python运行效率不高的问题
2020/07/20 Python
Python 用__new__方法实现单例的操作
2020/12/11 Python
利用html5 file api读取本地文件示例(如图片、PDF等)
2018/03/07 HTML / CSS
Html5调用手机摄像头并实现人脸识别的实现
2018/12/21 HTML / CSS
美国美妆网站:B-Glowing
2016/10/12 全球购物
英国工具中心:UK Tool Centre
2017/07/10 全球购物
Sneaker Studio波兰:购买运动鞋
2018/04/28 全球购物
阿里巴巴美国:Alibaba美国
2019/11/24 全球购物
全球最大的瓷器、水晶和银器零售商:Replacements
2020/06/15 全球购物
通信工程毕业生求职信
2013/11/16 职场文书
军人违纪检讨书
2014/02/04 职场文书
采购部经理岗位职责
2014/02/10 职场文书
团日活动总结范文
2014/04/25 职场文书
节能减耗标语
2014/06/21 职场文书