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 相关文章推荐
python2.7实现爬虫网页数据
May 25 Python
Python装饰器的执行过程实例分析
Jun 04 Python
使用python itchat包爬取微信好友头像形成矩形头像集的方法
Feb 21 Python
python中通过selenium简单操作及元素定位知识点总结
Sep 10 Python
详解Anconda环境下载python包的教程(图形界面+命令行+pycharm安装)
Nov 11 Python
如何基于python生成list的所有的子集
Nov 11 Python
解决Tensorflow 使用时cpu编译不支持警告的问题
Feb 03 Python
Python confluent kafka客户端配置kerberos认证流程详解
Oct 12 Python
关于python tushare Tkinter构建的简单股票可视化查询系统(Beta v0.13)
Oct 19 Python
python3 sqlite3限制条件查询的操作
Apr 07 Python
matplotlib之pyplot模块实现添加子图subplot的使用
Apr 25 Python
pytorch 权重weight 与 梯度grad 可视化操作
Jun 05 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
点评山进PR-D3L三波段收音机
2021/03/02 无线电
常用PHP框架功能对照表
2014/10/23 PHP
PHP 生成N个不重复的随机数
2015/01/21 PHP
PHP 生成微信红包代码简单
2016/03/25 PHP
PHP实现抽奖功能实例代码
2020/06/30 PHP
把textarea中字符串里含有的回车换行替换成<br>的javascript代码
2007/04/20 Javascript
Ext第一周 史上最强学习笔记---GridPanel(基础篇)
2008/12/29 Javascript
从URL中提取参数与将对象转换为URL查询参数的实现代码
2012/01/12 Javascript
jquery实现select选中行、列合计示例
2014/04/25 Javascript
js实现正方形颜色从下往上升的效果
2014/08/04 Javascript
node.js中的path.join方法使用说明
2014/12/08 Javascript
复杂的javascript窗口分帧解析
2016/02/19 Javascript
javascript移动开发中touch触摸事件详解
2016/03/18 Javascript
js判断所有表单项不为空则提交表单的实现方法
2016/09/09 Javascript
js实现小窗口拖拽效果
2016/12/03 Javascript
谈谈JavaScript数组常用方法总结
2017/01/24 Javascript
WdatePicker.js时间日期插件的使用方法
2017/07/26 Javascript
angularjs使用div模拟textarea文本框的方法
2018/10/02 Javascript
vuex实现的简单购物车功能示例
2019/02/13 Javascript
详解Angular cli配置过程记录
2019/11/07 Javascript
python写入已存在的excel数据实例
2018/05/03 Python
python tornado微信开发入门代码
2018/08/24 Python
python实现获取单向链表倒数第k个结点的值示例
2019/10/24 Python
Python Socketserver实现FTP文件上传下载代码实例
2020/03/27 Python
HTML5实现自带进度条和滑块滑杆效果
2018/04/17 HTML / CSS
What is the purpose of Void class? Void类的作用是什么?
2016/10/31 面试题
技校毕业生自荐信范文
2014/03/07 职场文书
文体活动实施方案
2014/03/27 职场文书
副护士长竞聘演讲稿
2014/04/30 职场文书
读书之星事迹材料
2014/05/12 职场文书
2014年国庆节广播稿
2014/09/19 职场文书
2014党的群众路线教育实践活动学习心得体会
2014/10/31 职场文书
校园环境卫生倡议书
2015/04/29 职场文书
小平小道观后感
2015/06/09 职场文书
利用Python网络爬虫爬取各大音乐评论的代码
2021/04/13 Python
Win11 vmware不兼容怎么办?Win11与VMware虚拟机不兼容的解决方法
2023/01/09 数码科技