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 struct.unpack
Sep 06 Python
解决uWSGI的编码问题详解
Mar 24 Python
Python语言描述连续子数组的最大和
Jan 04 Python
Python从使用线程到使用async/await的深入讲解
Sep 16 Python
python进行TCP端口扫描的实现
Dec 21 Python
Python读取xlsx文件的实现方法
Jul 04 Python
解决Atom安装Hydrogen无法运行python3的问题
Aug 28 Python
tensorflow如何批量读取图片
Aug 29 Python
django formset实现数据表的批量操作的示例代码
Dec 06 Python
Python类中self参数用法详解
Feb 13 Python
python爬虫开发之urllib模块详细使用方法与实例全解
Mar 09 Python
Docker如何部署Python项目的实现详解
Oct 26 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
Body是什么,该怎么喝出咖啡里的口感
2021/03/03 咖啡文化
frename PHP 灵活文件命名函数 frename
2009/09/09 PHP
php 代码优化的42条建议 推荐
2009/09/25 PHP
php调用C代码的实现方法
2014/03/11 PHP
php实现字符串反转输出的方法
2015/03/14 PHP
PHP实现冒泡排序的简单实例
2016/05/26 PHP
JS 页面内容搜索,类似于 Ctrl+F功能的实现代码
2007/08/13 Javascript
jQuery1.6 使用方法一
2011/11/23 Javascript
Prototype源码浅析 String部分(三)之HTML字符串处理
2012/01/15 Javascript
jquerymobile checkbox及时刷新才能获取其准确值
2012/04/14 Javascript
Angularjs制作简单的路由功能demo
2015/04/14 Javascript
用Node.js通过sitemap.xml批量抓取美女图片
2015/05/28 Javascript
IE8下jQuery改变png图片透明度时出现的黑边
2015/08/30 Javascript
js的各种排序算法实现(总结)
2016/07/23 Javascript
在Vue-cli里应用Vuex的state和mutations方法
2018/09/16 Javascript
JS立即执行函数功能与用法分析
2019/01/15 Javascript
为什么要使用Vuex的介绍
2019/01/19 Javascript
JavaScript Reflect Metadata实现详解
2019/12/12 Javascript
在vue中使用jsonp进行跨域请求接口操作
2020/10/29 Javascript
Python实现删除当前目录下除当前脚本以外的文件和文件夹实例
2015/07/27 Python
Python常见数据结构之栈与队列用法示例
2019/01/14 Python
深入理解Python异常处理的哲学
2019/02/01 Python
python 消除 futureWarning问题的解决
2019/12/25 Python
Python如何读写CSV文件
2020/08/13 Python
大韩航空官方网站:Korean Air
2017/10/25 全球购物
维多利亚的秘密官方旗舰店:VICTORIA’S SECRET
2018/04/02 全球购物
世界排名第一的万圣节服装店:Spirit Halloween
2018/10/16 全球购物
Manuka Doctor英国官网:真正的麦卢卡蜂蜜和护肤品
2018/10/26 全球购物
“四风”问题整改措施和努力方向
2014/09/20 职场文书
2014党委书记四风问题对照检查材料思想汇报
2014/09/22 职场文书
社区党的群众路线教育实践活动总结材料
2014/10/31 职场文书
婚礼新人答谢词
2015/01/04 职场文书
导游词欢迎词
2015/02/02 职场文书
怒海潜将观后感
2015/06/11 职场文书
《金色的草地》教学反思
2016/02/17 职场文书
正确使用MySQL update语句
2021/05/26 MySQL