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单链表简单实现代码
Apr 27 Python
python中print的不换行即时输出的快速解决方法
Jul 20 Python
Python实现批量检测HTTP服务的状态
Oct 27 Python
python3实现ftp服务功能(服务端 For Linux)
Mar 24 Python
Python解析命令行读取参数--argparse模块使用方法
Jan 23 Python
python代码过长的换行方法
Jul 19 Python
python使用epoll实现服务端的方法
Oct 16 Python
python 输出所有大小写字母的方法
Jan 02 Python
python使用html2text库实现从HTML转markdown的方法详解
Feb 21 Python
使用python绘制cdf的多种实现方法
Feb 25 Python
用Python简陋模拟n阶魔方
Apr 17 Python
python pygame 开发五子棋双人对弈
May 02 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
历史证明,懒惰才是推动科学发展技术进步的动力
2021/03/02 无线电
杏林同学录(二)
2006/10/09 PHP
一个PHP并发访问实例代码
2012/09/06 PHP
CI框架源码阅读,系统常量文件constants.php的配置
2013/02/28 PHP
Thinkphp的volist标签嵌套循环使用教程
2014/07/08 PHP
国产PHP开发框架myqee新手快速入门教程
2014/07/14 PHP
Alliance vs Liquid BO3 第一场2.13
2021/03/10 DOTA
Javascript动画的实现原理浅析
2015/03/02 Javascript
Jquery跨浏览器文本复制插件Zero Clipboard的使用方法
2016/02/28 Javascript
Bootstrap表单组件教程详解
2016/04/26 Javascript
jQuery Select下拉框操作小结(推荐)
2016/07/22 Javascript
使用get方式提交表单在地址栏里面不显示提交信息
2017/02/21 Javascript
jQuery中layer分页器的使用
2017/03/13 Javascript
js实现图片旋转 js滚动鼠标中间对图片放大缩小
2017/07/05 Javascript
JS实现下拉菜单列表与登录注册弹窗效果
2017/08/10 Javascript
vue-cli脚手架搭建的项目去除eslint验证的方法
2018/09/29 Javascript
mpvue+vant app搭建微信小程序的方法步骤
2019/02/11 Javascript
3分钟了解vue数据劫持的原理实现
2019/05/01 Javascript
vue实现列表滚动的过渡动画
2020/06/29 Javascript
[49:42]DOTA2上海特级锦标赛主赛事日 - 3 胜者组第二轮#2Secret VS EG第一局
2016/03/04 DOTA
[01:19:33]DOTA2-DPC中国联赛 正赛 iG vs VG BO3 第一场 2月2日
2021/03/11 DOTA
详解详解Python中writelines()方法的使用
2015/05/25 Python
python实现excel读写数据
2021/03/02 Python
python返回数组的索引实例
2019/11/28 Python
Python读取Excel数据并生成图表过程解析
2020/06/18 Python
详解pycharm自动import所需的库的操作方法
2020/11/30 Python
CSS类名支持中文命名的示例
2014/04/04 HTML / CSS
临床医学大学生求职信
2013/09/28 职场文书
小学生演讲稿
2014/01/12 职场文书
员工薪酬福利制度
2014/01/17 职场文书
幼儿园老师新年寄语2015
2014/12/08 职场文书
《一面五星红旗》教学反思
2016/02/23 职场文书
导游词之舟山普陀山
2019/11/06 职场文书
Python进度条的使用
2021/05/17 Python
Node-Red实现MySQL数据库连接的方法
2021/08/07 MySQL
Python语法学习之进程的创建与常用方法详解
2022/04/08 Python