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实现简单QQ单用户机器人的方法
Jul 03 Python
Python使用爬虫猜密码
Feb 19 Python
使用Python多线程爬虫爬取电影天堂资源
Sep 23 Python
python 定时任务去检测服务器端口是否通的实例
Jan 26 Python
Django调用支付宝接口代码实例详解
Apr 04 Python
500行python代码实现飞机大战
Apr 24 Python
django 数据库返回queryset实现封装为字典
May 19 Python
Pytorch转tflite方式
May 25 Python
Python创建临时文件和文件夹
Aug 05 Python
Python爬虫中Selenium实现文件上传
Dec 04 Python
如何理解python接口自动化之logging日志模块
Jun 15 Python
Python selenium绕过webdriver监测执行javascript
Apr 12 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
ThinkPHP中RBAC类的四种用法分析
2014/11/24 PHP
WIN8.1下搭建PHP5.6环境
2015/04/29 PHP
document.getElementById方法在Firefox与IE中的区别
2010/05/18 Javascript
JavaScript入门之事件、cookie、定时等
2011/10/21 Javascript
基于JavaScript自定义构造函数的详解说明
2013/04/24 Javascript
利用jq让你的div居中的好方法分享
2013/11/21 Javascript
nodejs文件操作模块FS(File System)常用函数简明总结
2014/06/05 NodeJs
node.js中的http.response.addTrailers方法使用说明
2014/12/14 Javascript
javascript设计简单的秒表计时器
2020/09/05 Javascript
jquery实现图片预加载
2015/12/25 Javascript
Javascript iframe交互并兼容各种浏览器的解决方法
2016/07/12 Javascript
如何使用Vuex+Vue.js构建单页应用
2016/10/27 Javascript
详解angular中通过$location获取路径(参数)的写法
2017/03/21 Javascript
详解基于node的前端项目编译时内存溢出问题
2017/08/01 Javascript
weex slider实现滑动底部导航功能
2017/08/28 Javascript
React Native 自定义下拉刷新上拉加载的列表的示例
2018/03/01 Javascript
vue服务端渲染缓存应用详解
2018/09/12 Javascript
angularJS1 url中携带参数的获取方法
2018/10/09 Javascript
JS实现简单移动端鼠标拖拽
2020/07/23 Javascript
Vue使用CDN引用项目组件,减少项目体积的步骤
2020/10/30 Javascript
python赋值操作方法分享
2013/03/23 Python
Python NumPy库安装使用笔记
2015/05/18 Python
Python函数的周期性执行实现方法
2016/08/13 Python
Python 异常处理的实例详解
2017/09/11 Python
python多线程共享变量的使用和效率方法
2019/07/16 Python
Python:合并两个numpy矩阵的实现
2019/12/02 Python
pandas factorize实现将字符串特征转化为数字特征
2019/12/19 Python
python使用建议与技巧分享(一)
2020/08/17 Python
Alba Moda德国网上商店:意大利时尚女装销售
2016/11/14 全球购物
MIKI HOUSE美国官方网上商店:日本领先的婴儿和儿童高级时装品牌
2020/06/21 全球购物
结婚典礼证婚词
2014/01/08 职场文书
电大本科自我鉴定
2014/02/05 职场文书
计算机多媒体专业自荐信
2014/07/04 职场文书
关于运动会的宣传稿
2015/07/23 职场文书
Python帮你解决手机qq微信内存占用太多问题
2022/02/15 Python
Linux中如何安装并部署Redis
2022/04/18 Servers