单例模式是最简单,也是最常用的一种设计模式,每种语言实现它的方式也有很多种。这里主要针对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))
一切完美解决。