python中单例常用的几种实现方法总结

作者:python修行路 时间:2022-02-28 19:37:55 

前言

最近这两天在看自己之前写的代码,所以正好把用过的东西整理一下,单例模式,在日常的代码工作中也是经常被用到,

所以这里把之前用过的不同方式实现的单例方式整理一下

什么是单例?

确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例,这个类称为单例类,单例模式是一种对象创建型模式。

那么单例模式有什么用途呢?举个常见的单例模式例子,我们平时使用的电脑上都有一个回收站,在整个操作系统中,回收站只能有一个实例,整个系统都使用这个唯一的实例,而且回收站自行提供自己的实例,因此回收站是单例模式的应用。

装饰器的方式

这种方式也是工作中经常用的一种,用起来也比较方便,代码实现如下


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

这样就很好的实现一个可以多线程使用的单例

来源:https://www.cnblogs.com/zhaof/p/9774525.html

标签:python,单例,常用
0
投稿

猜你喜欢

  • 详解如何使用Python网络爬虫获取招聘信息

    2021-09-28 06:58:17
  • python学习教程之Numpy和Pandas的使用

    2022-12-14 12:41:06
  • python简单猜数游戏实例

    2023-10-05 21:31:55
  • 如何在Pycharm中制作自己的爬虫代码模板

    2021-03-09 12:07:57
  • python中scipy.stats产生随机数实例讲解

    2021-03-20 07:57:04
  • 用python代码将tiff图片存储到jpg的方法

    2021-11-24 19:54:49
  • Python %r和%s区别代码实例解析

    2023-10-07 17:01:08
  • 解决MySQL 5数据库连接超时问题

    2009-03-25 15:24:00
  • python使用PyFetion来发送短信的例子

    2022-08-22 09:43:54
  • Python语言中的数据类型-序列

    2023-08-31 14:36:24
  • Mysql的服务无法启动的1067错误解决

    2012-01-05 19:31:56
  • PHP For循环字母A-Z当超过26个字母时输出AA,AB,AC

    2023-10-07 08:09:10
  • CSS压缩:技巧与工具

    2009-12-11 18:26:00
  • 基于Django框架的rest_framework的身份验证和权限解析

    2021-02-21 23:42:58
  • 彻底弄懂CSS盒子模式之四(绝对定位和相对定位)

    2007-05-11 16:51:00
  • SQL中exists的使用方法

    2011-12-01 08:36:07
  • python与字符编码问题

    2022-09-02 01:08:45
  • python实现决策树C4.5算法详解(在ID3基础上改进)

    2022-05-06 08:01:57
  • python递归函数绘制分形树的方法

    2021-04-22 02:16:02
  • SQL0290N表空间状态问题:停顿的独占处理

    2008-12-26 17:24:00
  • asp之家 网络编程 m.aspxhome.com