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算法学习之基数排序实例
Dec 18 Python
Python实现简单状态框架的方法
Mar 19 Python
pandas 按照特定顺序输出的实现代码
Jul 10 Python
Python中的groupby分组功能的实例代码
Jul 11 Python
Python实现输入二叉树的先序和中序遍历,再输出后序遍历操作示例
Jul 27 Python
Django unittest 设置跳过某些case的方法
Dec 26 Python
Python实现微信消息防撤回功能的实例代码
Apr 29 Python
pyqt 实现QlineEdit 输入密码显示成圆点的方法
Jun 24 Python
python垃圾回收机制(GC)原理解析
Dec 30 Python
Python读取配置文件(config.ini)以及写入配置文件
Apr 08 Python
Python confluent kafka客户端配置kerberos认证流程详解
Oct 12 Python
pytorch 带batch的tensor类型图像显示操作
May 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
解析PHP留言本模块主要功能的函数说明(代码可实现)
2013/06/25 PHP
php类常量用法实例分析
2015/07/09 PHP
php抛出异常与捕捉特定类型的异常详解
2016/10/26 PHP
JavaScript去掉空格的方法集合
2010/12/28 Javascript
JQuery的Ajax跨域请求原理概述及实例
2013/04/26 Javascript
Javascript全局变量var与不var的区别深入解析
2013/12/09 Javascript
JavaScript数值数组排序示例分享
2014/05/27 Javascript
node.js中的fs.writeSync方法使用说明
2014/12/15 Javascript
node.js中的fs.readlink方法使用说明
2014/12/17 Javascript
JavaScript动态修改网页元素内容的方法
2015/03/21 Javascript
分享我对JS插件开发的一些感想和心得
2016/02/04 Javascript
关于vue.js弹窗组件的知识点总结
2016/09/11 Javascript
ThinkPHP+jquery实现“加载更多”功能代码
2017/03/11 Javascript
vue学习笔记之vue1.0和vue2.0的区别介绍
2017/05/17 Javascript
vue学习笔记之过滤器的基本使用方法实例分析
2020/02/01 Javascript
vue vant中picker组件的使用
2020/11/03 Javascript
Python Web服务器Tornado使用小结
2014/05/06 Python
Python实现简单拆分PDF文件的方法
2015/07/30 Python
Python Sql数据库增删改查操作简单封装
2016/04/18 Python
python 常见字符串与函数的用法详解
2018/11/23 Python
在Python中如何传递任意数量的实参的示例代码
2019/03/21 Python
在pytorch中为Module和Tensor指定GPU的例子
2019/08/19 Python
Python利用PyPDF2库获取PDF文件总页码实例
2020/04/03 Python
Sql面试题
2013/03/20 面试题
某公司的.net工程师面试题笔试题
2013/11/22 面试题
新闻专业推荐信范文
2013/11/20 职场文书
求职信需要的五点内容
2014/02/01 职场文书
数控专业毕业生求职信
2014/06/12 职场文书
“三支一扶”支教教师思想汇报
2014/09/13 职场文书
2014县委书记四风对照检查材料思想汇报
2014/09/21 职场文书
2014年资料员工作总结
2014/11/18 职场文书
捐书活动倡议书
2015/04/27 职场文书
文艺晚会开场白
2015/05/29 职场文书
2016秋季运动会开幕词
2016/03/04 职场文书
正则表达式拆分url实例代码
2022/02/24 Java/Android
html解决浏览器记住密码输入框的问题
2023/05/07 HTML / CSS