第21章:单例模式
单例模式
单例模式(Singleton):保证一个类仅有一个实例, 并提供一个访问它的全局访问点。
通常可以用一个全局变量使一个对象被访问,但它不能防止实例化多个对象。一个最好的办法是让类自身负责保存它的唯一实例。这个类可以保证没有其它实例被创建,并且它可以提供一个访问该实例的方法。
Singleton
类,定义一个Getlnstance
操作,允许客户访问它的唯一实例。GetInstance
是一个静态方法,主要负责创建自己的唯一实例。
客户端代码
由于Singleton
类封装它的唯一实例,这样它可以严格地控制客户怎样访问它以及何时访问它。简单说就是对唯一实例的受控访问。
单例类
与实用类
的区别
-
实用类通常也会采用私有化的构造方法来避免其有实例,但实用类不保存状态,仅提供一些静态方法或静态属性;而单例类是有状态的。
-
实用类不能用于继承多态,而单例虽然实例唯一,却是可以有子类来继承。
-
实用类是一些方法属性的集合,而单例却是有着唯一的对象实例。
单例模式示例
class Singleton(object):
def __init__(self, cls: object) -> None:
self._cls = cls
self._instance = {
}
def __call__(self, *args, **kwargs) -> object:
if self._cls not in self._instance:
self._instance[self._cls] = self._cls(*args, **kwargs)
return self._instance[self._cls]
@Singleton
class ConcreteSingleton(object):
def __init__(self) -> None:
pass
if __name__ == "__main__":
single1 = ConcreteSingleton()
single2 = ConcreteSingleton()
print(id(single1) == id(single2))
True
多线程时的单例
多线程的程序中,多个线程同时访问Singleton
类(注意是同时),调用GetInstance()
方法,有可能会创建多个实例。此时,可以通过给进程一把锁来处理。
lock
语句的涵义:lock
是确保当一个线程位于代码的临界区时,另一个线程不进入临界区。如果其他线程试图进入锁定的代码,则它将一直等待(即被阻止),直到该对象被释放。
from threading import Lock
class Singleton(object):
def __init__(self, cls: object) -> None:
self._cls = cls
self._instance = {
}