Python实现Singleton模式的几种方式
参考:https://www.cnblogs.com/yssjun/p/9858420.html
https://blog.csdn.net/zzddada/article/details/113771933
使用python实现设计模式中的单例模式。单例模式是一种比较常用的设计模式,其实现和使用场景判定都是相对容易的。本文将简要介绍一下python中实现单例模式的几种常见方式和原理。一方面可以加深对python的理解,另一方面可以更加深入的了解该模式,以便实际工作中能更加灵活的使用单例设计模式。
本文将介绍常见的实现单例模式的几种方式,这里暂不考虑多线程的情况。
为了准备该篇博文,之前写了几篇相关的文章依次完整的介绍了相关的概念,下面会在需要的时候给出链接。
装饰器作为python实现单例模式的一种常用方法,先简单了解一下其概念。
1.装饰器
装饰器(Decorator)可以用作对函数以及类进行二次包裹或者封装,使用方式@wrapper。
def f(...):
...
f = staticmethod(f)
@staticmethod
def f(...):
...
上面这两种方式对函数的定义在语法上是等价的。当然对于类也有同样的用法,类可以作为装饰器也可以作为被装饰对象。唯一的区别就是经过包裹的类可能不在是一个类,而是一个类的对象或者一个函数,这取决于装饰器返回的值。
经过Decorator装饰的类或者函数本质上已经不再是原来的类或者函数了。但是,实际上在包裹之后得到的新对象仍然拥有被包裹对象的特性(这句是不是废话:-))。
在python中我们经常只需要实现一个装饰器,然后使用该装饰器作用于只能有唯一一个实例的类。这样只需要实现一个这样的装饰器,便可以作用于任何一个想要唯一实例的类。
2.闭包方式
闭包的应用很多,单例模式则是其应用之一。先看代码:
1 def singleton(cls):
2 instances = {}
3
4 def getinstance(*args, **kwargs):
5 if cls not in instances:
6 instances[cls] = cls(*args, **kwargs)
7 return instances[cls]
8 return getinstance
9
10 @singleton
11 class my_cls(object):
12 pass
这个实现单例模式的方式将原来类的定义隐藏在闭包函数中,通过闭包函数及其中引用的自由变量来控制类对象的生成。由于唯一的实例存放在自由变量中,而且自由变量是无法直接在脚本层进行访问的。这种方式非常隐蔽的保护实例不被修改,因此很适合用于单例模式。
这种方式简单明了,很容易实现。但是如果不了解闭包实现过程和变量的绑定等概念可能会不明白其实现的过程。建议参考这篇博文:理解python闭包概念。
这里一个很有趣的地方是为什么要使用instances = {}这样一个变量?可不可以不用字典,使用instance = None?如果singleton作为装饰器被多个不同的类使用,那么instance中会存在几个不同的实例么?
有时间可以思考一下这几个问题,答案也可以在闭包相关的博文中找到。