python 设计模式-单例模式

单例模式是最简单,也是最常用的一种设计模式,每种语言实现它的方式也有很多种。这里主要针对python中的实现进行说明,个人觉得使用重写__new__的方式来实现单例是最优雅的。

先看下面这个例子:

class Singleton:
    __instance = None
    
    def __new__(cls, *args, **kwargs):
        if cls.__instance is None:
            cls.__instance = super(Singleton, cls).__new__(cls, *args, **kwargs)
        return cls.__instance


if __name__ == '__main__':
    s1 = Singleton()
    s2 = Singleton()
    print("s1:{}, s2:{}".format(s1, s2))

输出

s1:<__main__.Singleton object at 0x03A8CED0>, s2:<__main__.Singleton object at 0x03A8CED0>

重写__new__魔术方法,很容易的实现了单例模式。但是针对每一个需要实现单例的类都要重写一下这个函数,有没有更好的方式呢?看如下示例:

class SingletonBase:
    __instance = {}
    
    def __new__(cls, *args, **kwargs):
        return cls.__instance.setdefault(cls, super(SingletonBase, cls).__new__(cls, *args, **kwargs))
    
         
class Singleton(SingletonBase):            
    pass


if __name__ == '__main__':
    s1 = Singleton()
    s2 = Singleton()
    print("s1:{}, s2:{}".format(s1, s2))

实现一个单例的基类,任何想成为单例的类只要继承它就可以了,又进了一步,但是这里有个优化的问题,一旦单例生成之后,它就会存在于基类的 __instance 类属性中,即使运行环境超出实例作用域,也无法被垃圾回收掉。这个时候,weakref 模块就排上用场了,修改下例子:

from weakref import WeakValueDictionary


class SingletonBase:
    __instance = WeakValueDictionary()
    
    def __new__(cls, *args, **kwargs):
        return cls.__instance.setdefault(cls, super(SingletonBase, cls).__new__(cls, *args, **kwargs))
    
         
class Singleton(SingletonBase):            
    pass


if __name__ == '__main__':
    s1 = Singleton()
    s2 = Singleton()
    print("s1:{}, s2:{}".format(s1, s2))

一切完美解决。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值