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 相关文章推荐
详解Django通用视图中的函数包装
Jul 21 Python
Python 正则表达式的高级用法
Dec 04 Python
python多进程读图提取特征存npy
May 21 Python
django一对多模型以及如何在前端实现详解
Jul 24 Python
Python操作SQLite/MySQL/LMDB数据库的方法
Nov 07 Python
python 使用raw socket进行TCP SYN扫描实例
May 05 Python
Python使用sqlite3模块内置数据库
May 07 Python
matplotlib.pyplot.matshow 矩阵可视化实例
Jun 16 Python
python如何导入依赖包
Jul 13 Python
Python连接Impala实现步骤解析
Aug 04 Python
解决Python安装cryptography报错问题
Sep 03 Python
用Python提取PDF表格的方法
Apr 11 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
4月1日重磅发布!《星际争霸II》6.0.0版本更新
2020/04/09 星际争霸
用PHP连mysql和oracle数据库性能比较
2006/10/09 PHP
PHP4引用文件语句的对比
2006/10/09 PHP
一个php生成16位随机数的代码(两种方法)
2014/09/16 PHP
php文件扩展名判断及获取文件扩展名的N种方法
2015/09/12 PHP
Symfony的安装和配置方法
2016/03/17 PHP
Yii视图CGridView实现操作按钮定义地址示例
2016/07/14 PHP
基于PHP实现用户注册登录功能
2016/10/14 PHP
setTimeout和setInterval的深入理解
2013/11/08 Javascript
JavaScript在Android的WebView中parseInt函数转换不正确问题解决方法
2015/04/25 Javascript
javascript动态设置样式style实例分析
2015/05/13 Javascript
JavaScript中的pow()方法使用详解
2015/06/15 Javascript
TypeScript 学习笔记之基本类型
2015/06/19 Javascript
js实现当复选框选择匿名登录时隐藏登录框效果
2015/08/14 Javascript
bootstrap布局中input输入框右侧图标点击功能
2016/05/16 Javascript
用JavaScript动态建立或增加CSS样式表的实现方法
2016/05/20 Javascript
ZeroClipboard.js使用一个flash复制多个文本框
2017/06/19 Javascript
jQuery实现获取table中鼠标click点击位置行号与列号的方法
2017/10/09 jQuery
详解vue-cli 3.0 build包太大导致首屏过长的解决方案
2018/11/10 Javascript
js如何获取访问IP、地区、当前操作浏览器
2019/07/23 Javascript
json字符串对象转换代码实例
2019/09/28 Javascript
vue addRoutes路由动态加载操作
2020/08/04 Javascript
多种类型jQuery网页验证码插件代码实例
2021/01/09 jQuery
Python中使用pprint函数进行格式化输出的教程
2015/04/07 Python
python简单猜数游戏实例
2015/07/09 Python
Python的自动化部署模块Fabric的安装及使用指南
2016/01/19 Python
Python实现八大排序算法
2016/08/13 Python
python多线程并发及测试框架案例
2019/10/15 Python
TensorFlow 多元函数的极值实例
2020/02/10 Python
Django admin管理工具TabularInline类用法详解
2020/05/14 Python
python 如何设置守护进程
2020/10/29 Python
Selenium关闭INFO:CONSOLE提示的解决
2020/12/07 Python
澳大利亚汽车零部件、音响及配件超市:Automotive Superstore
2018/06/19 全球购物
干部鉴定材料
2014/05/18 职场文书
TensorFlow的自动求导原理分析
2021/05/26 Python
详解CSS3浏览器兼容
2022/12/24 HTML / CSS