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装饰器在Django框架下去除冗余代码的教程
Apr 16 Python
用Python编写一个每天都在系统下新建一个文件夹的脚本
May 04 Python
详解Python装饰器由浅入深
Dec 09 Python
Python将多个excel文件合并为一个文件
Jan 03 Python
Python实现PS图像调整颜色梯度效果示例
Jan 25 Python
Python socket模块实现的udp通信功能示例
Apr 10 Python
Python 监测文件是否更新的方法
Jun 10 Python
python求最大值最小值方法总结
Jun 25 Python
Python 函数用法简单示例【定义、参数、返回值、函数嵌套】
Sep 20 Python
Pandas的Apply函数具体使用
Jul 21 Python
filter使用python3代码进行迭代元素的实例详解
Dec 03 Python
在pycharm中无法import所安装的库解决方案
May 31 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做的端口嗅探器--可以指定网站和端口
2006/10/09 PHP
用PHP的ob_start();控制您的浏览器cache!
2007/02/14 PHP
有关phpmailer的详细介绍及使用方法
2013/01/28 PHP
php中return的用法实例分析
2015/02/28 PHP
php文件上传、下载和删除示例
2020/08/28 PHP
PHP高效获取远程图片尺寸和大小的实现方法
2017/10/20 PHP
PHP实现单条sql执行多个数据的insert语句方法
2019/10/11 PHP
YII2框架中添加自定义模块的方法实例分析
2020/03/18 PHP
PHP使用Http Post请求发送Json对象数据代码解析
2020/07/16 PHP
div+css布局的图片连续滚动js实现代码
2010/05/04 Javascript
jQuery实现三级菜单的代码
2016/05/09 Javascript
JavaScript禁止复制与粘贴的实现代码
2016/05/16 Javascript
浅谈String.valueOf()方法的使用
2016/06/06 Javascript
Bootstrap table两种分页示例
2016/12/23 Javascript
nodeJS(express4.x)+vue(vue-cli)构建前后端分离实例(带跨域)
2017/07/05 NodeJs
Vue中的ref作用详解(实现DOM的联动操作)
2017/08/21 Javascript
微信小程序实现图片懒加载的示例代码
2017/12/13 Javascript
详解Vue SSR( Vue2 + Koa2 + Webpack4)配置指南
2018/11/13 Javascript
防止Layui form表单重复提交的实现方法
2019/09/10 Javascript
vue项目中使用vue-layer弹框插件的方法
2020/03/11 Javascript
Vue 组件的挂载与父子组件的传值实例
2020/09/02 Javascript
一篇文章搞懂Python的类与对象名称空间
2018/12/10 Python
Python面向对象程序设计OOP入门教程【类,实例,继承,重载等】
2019/01/05 Python
Python爬虫动态ip代理防止被封的方法
2019/07/07 Python
详解python调用cmd命令三种方法
2019/07/08 Python
Python socket模块ftp传输文件过程解析
2019/11/05 Python
python 中的paramiko模块简介及安装过程
2020/02/29 Python
Python日志器使用方法及原理解析
2020/09/27 Python
CSS教程:CSS3圆角属性
2009/04/02 HTML / CSS
BASIC HOUSE官方旗舰店:韩国著名的服装品牌
2018/09/27 全球购物
大学生村官任职感言
2014/01/09 职场文书
质量月口号
2014/06/20 职场文书
关于运动会的广播稿
2014/09/22 职场文书
某集团股份有限公司委托书样本
2014/09/24 职场文书
自主招生自荐信范文
2015/03/04 职场文书
【海涛解说】暗牧也疯狂,牛蛙成配角
2022/04/01 DOTA