单例python_小结python单例的两种实现方法

本文介绍了Python中实现单例模式的两种常见方法:装饰器方式和类方式。装饰器方式简洁易用,类方式则需要注意多线程环境下的线程安全问题。在类方式中,通过加锁解决了线程安全问题,但初始加锁操作可优化为仅在实例未创建时获取锁。最后给出了完整的代码示例,帮助读者理解并应用Python单例模式。
摘要由CSDN通过智能技术生成

python单例的两种实现方法介绍(附代码)

本篇文章给大家带来的内容是关于python单例的两种实现方法介绍(附代码),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。

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

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

装饰器的方式

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

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视频教程和python文章教程栏目!!!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值