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模块restful使用方法实例
Dec 10 Python
Python编程中运用闭包时所需要注意的一些地方
May 02 Python
一个基于flask的web应用诞生 flask和mysql相连(4)
Apr 11 Python
从django的中间件直接返回请求的方法
May 30 Python
可能是最全面的 Python 字符串拼接总结【收藏】
Jul 09 Python
Python实现html转换为pdf报告(生成pdf报告)功能示例
May 04 Python
关于Python作用域自学总结
Jun 10 Python
python 日期排序的实例代码
Jul 11 Python
Python全栈之列表数据类型详解
Oct 01 Python
Python 内置变量和函数的查看及说明介绍
Dec 25 Python
python处理RSTP视频流过程解析
Jan 11 Python
Python3.9 beta2版本发布了,看看这7个新的PEP都是什么
Jun 10 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
在WIN98下以apache模块方式安装php
2006/10/09 PHP
允许phpmyadmin空密码登录的配置方法
2011/05/29 PHP
PHP的简易冒泡法代码分享
2012/08/28 PHP
php格式化时间戳显示友好的时间实现思路及代码
2014/10/23 PHP
PHP闭包函数传参及使用外部变量的方法
2016/03/15 PHP
ppk谈JavaScript style属性
2008/10/10 Javascript
JS 修改URL参数(实现代码)
2013/07/08 Javascript
在js文件中写el表达式取不到值的原因及解决方法
2013/12/23 Javascript
JS+CSS实现Div弹出窗口同时背景变暗的方法
2015/03/04 Javascript
Javascript连接Access数据库完整实例
2015/08/03 Javascript
JS实现IE状态栏文字缩放效果代码
2015/10/24 Javascript
基于JavaScript操作DOM常用的API小结
2015/12/01 Javascript
JavaScript继承学习笔记【新手必看】
2016/05/10 Javascript
在一个页面重复使用一个js函数的方法详解
2016/12/26 Javascript
浅谈原生JS实现jQuery的animate()动画示例
2017/03/08 Javascript
详解vue 2.6 中 slot 的新用法
2019/07/09 Javascript
vue 动态表单开发方法案例详解
2019/12/02 Javascript
解决vuex改变了state的值,但是页面没有更新的问题
2020/11/12 Javascript
Python计算字符宽度的方法
2016/06/14 Python
Python+OpenCV人脸检测原理及示例详解
2020/10/19 Python
获取python的list中含有重复值的index方法
2018/06/27 Python
python2.7实现复制大量文件及文件夹资料
2019/08/31 Python
在Python IDLE 下调用anaconda中的库教程
2020/03/09 Python
使用matplotlib的pyplot模块绘图的实现示例
2020/07/12 Python
HTML5时代CSS设置漂亮字体取代图片
2014/09/04 HTML / CSS
一款CSS3实现多功能下拉菜单(带分享按)的教程
2014/11/05 HTML / CSS
css3的transform中scale缩放详解
2014/12/08 HTML / CSS
Vuori官网:运动服装的终级表现
2021/01/27 全球购物
结构和类有什么异同
2012/07/16 面试题
生育关怀行动实施方案
2014/03/26 职场文书
医学求职自荐信
2014/06/21 职场文书
公司演讲稿开场白
2014/08/25 职场文书
亮剑观后感500字
2015/06/05 职场文书
创业计划书之校园超市
2019/09/12 职场文书
比较node.js和Deno
2021/04/27 Javascript
MySQL一些常用高级SQL语句
2021/07/03 MySQL