常见的在Python中实现单例模式的三种方法
单例模式是一种常用的软件设计模式。在它的核心结构中只包含一个被称为单例类的特殊类。通过单例模式可以保证系统中一个类只有一个实例而且该实例易于外界访问,从而方便对实例个数的控制并节约系统资源。如果希望在系统中某个类的对象只能存在一个,单例模式是最好的解决方案。
单例模式的要点有三个;一是某个类只能有一个实例;二是它必须自行创建这个实例;三是它必须自行向整个系统提供这个实例。在Python中,单例模式有以下几种实现方式。
方法一、实现__new__方法,然后将类的一个实例绑定到类变量_instance上;如果cls._instance为None,则说明该类还没有被实例化过,new一个该类的实例,并返回;如果cls._instance不为None,直接返回_instance,代码如下:
classSingleton(object): def__new__(cls,*args,**kwargs): ifnothasattr(cls,'_instance'): orig=super(Singleton,cls) cls._instance=orig.__new__(cls,*args,**kwargs) returncls._instance classMyClass(Singleton): a=1 one=MyClass() two=MyClass() #one和two完全相同,可以用id(),==,is检测 printid(one)#29097904 printid(two)#29097904 printone==two#True printoneistwo#True
方法二、本质上是方法一的升级版,使用__metaclass__(元类)的高级python用法,具体代码如下:
classSingleton2(type): def__init__(cls,name,bases,dict): super(Singleton2,cls).__init__(name,bases,dict) cls._instance=None def__call__(cls,*args,**kwargs): ifcls._instanceisNone: cls._instance=super(Singleton2,cls).__call__(*args,**kwargs) returncls._instance classMyClass2(object): __metaclass__=Singleton2 a=1 one=MyClass2() two=MyClass2() printid(one)#31495472 printid(two)#31495472 printone==two#True printoneistwo#True
方法三、使用Python的装饰器(decorator)实现单例模式,这是一种更Pythonic的方法;单利类本身的代码不是单例的,通装饰器使其单例化,代码如下:
defsingleton(cls,*args,**kwargs): instances={} def_singleton(): ifclsnotininstances: instances[cls]=cls(*args,**kwargs) returninstances[cls] return_singleton @singleton classMyClass3(object): a=1 one=MyClass3() two=MyClass3() printid(one)#29660784 printid(two)#29660784 printone==two#True printoneistwo#True