python 单例模式

本文介绍了单例模式的概念及其在Python中的实现,包括饿汉式、懒汉式和双重检测。饿汉式在程序加载时即创建实例,适合资源管理;懒汉式延迟加载,适用于不频繁使用的场景;双重检测结合两者优点,支持高并发。文章总结了各种实现方式的优缺点,并指出在实际应用中,饿汉式更为常见。
摘要由CSDN通过智能技术生成

单例模式(Singleton Design Pattern)理解起来非常简单,一个类只允许创建一个实例,这个类就属于一个单例类。

适用范围

  1. 处理共享资源的访问冲突。
  2. 需要设置全局唯一类,比如日志类、数据库连接类、全局配置文件类等。
  3. 需要频繁实例化然后销毁的对象。

单例类的分类

需要注意的地方:

单例模式在 多线程 必须小心使用。如果当唯一实例尚未创建时,有两个线程同时调用创建方法,那么它们同时没有检测到唯一实例的存在,从而同时各自创建了一个实例, 这样就有两个实例被构造出来,从而违反了单例模式中实例唯一的原则。 解决这个问题的办法是为指示类是否已经实例化的变量提供一个互斥锁(虽然这样会降低效率)。

饿汉式

饿汉式:当一个类在引入或加载时,实例就创建并初始化。

直接在模块中生成类的实例对象:

class Log:
    pass

# 其他模块通过 import log 使用实例
log = Log()

优点

  1. 提前创建好实例,避免在程序运行的时候,再去初始化导致的性能问题。
  2. 按照 fail-fast 的设计原则(有问题及早暴露),有问题时初始化就能暴露(比如内存溢出),不用等到线上运行时,影响系统可用性。
  3. 线程安全的

缺点

  1. 初始化资源耗时,如果需要初始化的对象很多,程序加载时间太久,体验不好。
  2. 资源效率不高,实例初始化出来后可能一直没有使用到,造成浪费

懒汉式

懒汉式:延迟加载,只有在实例需要使用时,才初始化实例。

懒汉式在python有几种使用方法:

  • 函数装饰器
def borg(cls):
    """
    通过函数装饰器来实现一个类
    """
    _instance = dict()

    def singleton(*args, **kwargs):
        if cls not in _instance:
            _instance[cls] = cls(*args, **kwargs)
        return _instance[cls]

    return singleton


@borg
class Log1:
    def __init__(self):
        print("aaa")
  • 类装饰器
class Borg:
    def __init__(self, cls):
        self._cls = cls
        self._instance = {}

    def __call__(self):
        if self._cls not in self._instance:
            self._instance[self._cls] = self._cls
        return self._instance[self._cls]


@Borg
class Log2:
    def __init__(self):
        print("aaa")
  • 通过 python 的 new 方法
class Log:
    """
    通过 __new__ 方法实现一个单例类
    通过一个静态变量 _instance 用来保证实例唯一性
    """
    _instance = None

    def __new__(cls, *args, **kwargs):
        if not hasattr(cls, "_instance"):
            cls._instance = super().__new__(cls)
        return cls._instance

优缺点与上面饿汉式正好相反

双重检测

饿汉式不支持延迟加载,懒汉式有性能问题,不支持高并发。有一种既支持延迟加载、又支持高并发的单例实现方式,也就是双重检测实现方式。

通过双重检测方式,我们要保证同一时刻,多个线程只有一个能够操作实例的创建,我们可以通过多线程锁来实现。只要实例被创建之后,即便再调用 get_instance 函数也不会再进入到加锁逻辑中了
具体实例如下:

from threading import Lock


class Log4:
    """
    通过一个多线程锁保证同一时刻最多只有一个线程能创建实例
    不能通过 new 方法创建实例
    """
    _lock = Lock()

    def __new__(cls, *args, **kwargs):
        msg = "Cann't new a SingletonSample class, please use get_instance method!"
        raise NotImplementedError(msg)

    @classmethod
    def get_instance(cls):
        if not hasattr(cls, "_instance"):
            with cls._lock:
                if not hasattr(cls, "_instance"):
                    cls._instance = super().__new__(cls)
        return cls._instance

总结

这篇文章主要总结了一下自己学习 单例模式 时学到的内容,一般 单例模式 都是采用饿汉式,比较少采用懒汉式及双重检测。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

tlqwanttolearnit

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值