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通过yield实现数组全排列的方法
Mar 18 Python
在Python中使用全局日志时需要注意的问题
May 06 Python
Python字符串格式化
Jun 15 Python
最大K个数问题的Python版解法总结
Jun 16 Python
python多进程提取处理大量文本的关键词方法
Jun 05 Python
解决Shell执行python文件,传参空格引起的问题
Oct 30 Python
使用python serial 获取所有的串口名称的实例
Jul 02 Python
Python Numpy数组扩展repeat和tile使用实例解析
Dec 09 Python
Python模块的定义,模块的导入,__name__用法实例分析
Jan 07 Python
python飞机大战游戏实例讲解
Dec 04 Python
scrapy头部修改的方法详解
Dec 06 Python
Python爬虫回测股票的实例讲解
Jan 22 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
DC动画电影《黑暗正义联盟》曝预告 5月5日上线数字平台
2020/04/09 欧美动漫
php长字符串定义方法
2012/07/12 PHP
PHP面向对象精要总结
2014/11/07 PHP
php文件操作相关类实例
2015/06/18 PHP
PHP处理会话函数大总结
2015/08/05 PHP
详解PHP中foreach的用法和实例
2016/10/25 PHP
jQuery中bind()方法用法实例
2015/01/19 Javascript
js实现按钮颜色渐变动画效果
2015/08/20 Javascript
jQuery多级手风琴菜单实例讲解
2015/10/22 Javascript
js文本框输入内容智能提示效果
2015/12/02 Javascript
基于BootStrap的前端分页带省略号和上下页效果
2017/05/18 Javascript
Vue的移动端多图上传插件vue-easy-uploader的示例代码
2017/11/27 Javascript
浅谈React深度编程之受控组件与非受控组件
2017/12/26 Javascript
vue-cli3配置与跨域处理方法
2019/08/17 Javascript
基于JS实现table导出Excel并保留样式
2020/05/19 Javascript
vue实现简单学生信息管理
2020/05/30 Javascript
从零学Python之入门(五)缩进和选择
2014/05/27 Python
Python对列表排序的方法实例分析
2015/05/16 Python
python日期时间转为字符串或者格式化输出的实例
2018/05/29 Python
PyQt5 窗口切换与自定义对话框的实例
2019/06/20 Python
python 中xpath爬虫实例详解
2019/08/26 Python
python对象销毁实例(垃圾回收)
2020/01/16 Python
Python数据结构dict常用操作代码实例
2020/03/12 Python
python中数字是否为可变类型
2020/07/08 Python
eBay法国购物网站:eBay.fr
2017/10/21 全球购物
一家专门经营包包的英国网站:MyBag
2019/09/08 全球购物
俄罗斯最大的在线珠宝大卖场:Nebo
2019/12/08 全球购物
.NET初级开发工程师面试题(包括Javascript)
2012/08/22 面试题
面向对象设计的原则是什么
2013/02/13 面试题
终止或解除劳动合同及劳动关系的证明书
2014/10/06 职场文书
幼儿园家长心得体会
2016/01/21 职场文书
八年级作文之友情
2019/11/25 职场文书
拒绝盗图!教你怎么用python给图片加水印
2021/06/04 Python
MySQL空间数据存储及函数
2021/09/25 MySQL
PyTorch device与cuda.device用法
2022/04/03 Python
vue-cli3.x配置全局的scss的时候报错问题及解决
2022/04/30 Vue.js