python单例模式原理是什么_python 单例模式的四种实现方法及注意事项

一、模块单例

Python 的模块就是天然的单例模式,因为模块在第一次导入时,会生成 .pyc 文件,当第二次导入时,就会直接加载 .pyc 文件,而不会再次执行模块代码。

#foo1.py

classSingleton(object):deffoo(self):passsingleton=Singleton()#foo.py

from foo1 import singleton

直接在其他文件中导入此文件中的对象,这个对象即是单例模式的对象

二、静态变量方法

先执行了类的__new__方法(我们没写时,默认调用object.__new__),实例化对象;然后再执行类的__init__方法,对这个对象进行初始化,所有我们可以基于这个,实现单例模式。

classSingleton(object):def __new__(cls,a):if not hasattr(cls, '_instance'):

cls._instance= object.__new__(cls)returncls._instancedef __init__(self,a):

self.a=adefaa(self):print(self.a)

a= Singleton("a")

变种:利用类的静态方法或者类方法,实现对函数初始化的控制。该方法需要手动调用静态方法实现实例。本质上是手动版的__new__方法。

ContractedBlock.gif

ExpandedBlockStart.gif

classSingleton:

@staticmethoddef instance1(*args, **kwargs):if not hasattr(Singleton, "_instance"):

Singleton._instance= Singleton(*args, **kwargs)returnSingleton._instance

@classmethoddef instance2(cls,*args, **kwargs):if not hasattr(cls, "_instance"):

Singleton._instance= Singleton(*args, **kwargs)returnSingleton._instance

a=Singleton.instance1()

b=Singleton.instance2()print(a == b)

手动版静态版

三、元类方法

此方法是在__new__方法的更上层对实例化过程进行控制。

原理:执行元类的 元类的__new__方法和__init__方法用来实例化类对象,__call__ 方法用来对实例化的对象的实例即类的对象进行控制。__call__方法会调用实例类的 __new__方法,用于创建对象。返回对象给__call__方法,然后调用类对象的 __init__方法,用于对对象初始化。

classSingleton1(type):def __init__(self, *args, **kwargs):

self.__instance =None

super(Singleton1,self).__init__(*args, **kwargs)def __call__(self, *args, **kwargs):if self.__instance isNone:

self.__instance = super(Singleton1,self).__call__(*args, **kwargs)return self.__instance

classSingleton2(type):

_inst={}def __call__(cls, *args, **kwargs):print(cls)if cls not incls._inst:

cls._inst[cls]= super(Singleton2, cls).__call__(*args)returncls._inst[cls]class C(metaclass=Singleton1):pass

四、装饰器

原理:装饰器用来控制类调用__call__方法。

def singleton(cls, *args, **kw):

instance={}def_singleton(args):if cls not ininstance:

instance[cls]= cls(*args, **kw)returninstance[cls]return_singleton

@singletonclassA:pass

五、注意:

除了模块单例外,其他几种模式的本质都是通过设置中间变量,来判断类是否已经被实例。区别就是中间变量的位置不同,或设置在元类中,或封装在函数中,或设置在类中作为静态变量。

注意1:中间变量的访问和更改存在线程安全的问题:在开启多线程模式的时候需要加锁处理。

注意2:__new__方法无法避免触发__init__(),初始的成员变量会进行覆盖。其他方法不会。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值