Python 用__new__方法實(shí)現(xiàn)單例的操作
介紹
init 方法通常用在初始化一個(gè)類(lèi)實(shí)例時(shí)候,但其實(shí)它不是實(shí)例化一個(gè)類(lèi)的時(shí)候第一個(gè)被調(diào)用 的方法。當(dāng)使用 Student(id, name) 這樣的表達(dá)式來(lái)實(shí)例化一個(gè)類(lèi)時(shí),最先被調(diào)用的方法 其實(shí)是 new 方法。
new方法接受的參數(shù)雖然也是和init一樣,但init是在類(lèi)實(shí)例創(chuàng)建之后調(diào)用,而 new方法正是創(chuàng)建這個(gè)類(lèi)實(shí)例的方法。
new為對(duì)象分配空間,是內(nèi)置的靜態(tài)方法,new在內(nèi)存中為對(duì)象分配了空間也返回了對(duì)象的引用,init獲得了這個(gè)引用才初始化這個(gè)實(shí)例。
示例
一個(gè)非常簡(jiǎn)單的單例
class A: instance = None def __new__(cls, *args, **kwargs): if cls.instance is None: cls.instance = super().__new__(cls) return cls.instance
因?yàn)閚ew方法是一個(gè)靜態(tài)方法(也就是在定義的時(shí)候就沒(méi)有cls參數(shù)),所以在這里要傳入一個(gè)cls參數(shù),而且這里的new你改造過(guò)了,所以要返回爸爸的new方法。
按造這個(gè)方法改造的單例怎么new都是同一個(gè)實(shí)例,但init仍然會(huì)被執(zhí)行多次,也就是創(chuàng)建了幾個(gè)對(duì)象就調(diào)用幾次初始化方法。所以還要對(duì)init再進(jìn)行一些判斷。
class A: instance = None init_flag = False # 初始化標(biāo)記 def __new__(cls, *args, **kwargs): if cls.instance is None: cls.instance = super().__new__(cls) return cls.instance def __init__(self): if A.init_flag: return print(’執(zhí)行了初始化方法’) A.init_flag = Trueif __name__ == ’__main__’: a = A() b = A() print(a) print(b)
輸出結(jié)果:
執(zhí)行了初始化方法
<main.A object at 0x00000210E6F09320>
<main.A object at 0x00000210E6F09320>
總結(jié)
通過(guò)重載new方法,可以比較簡(jiǎn)單地實(shí)現(xiàn)單例,Python還有很多有趣的內(nèi)置函數(shù),有空可以再研究研究。
補(bǔ)充知識(shí):Python餓漢式和懶漢式單例模式的實(shí)現(xiàn)
看代碼吧~
# 餓漢式class Singleton(object): # 重寫(xiě)創(chuàng)建實(shí)例的__new__方法 def __new__(cls): # 如果類(lèi)沒(méi)有實(shí)例屬性,進(jìn)行實(shí)例化,否則返回實(shí)例 if not hasattr(cls, ’instance’): cls.instance = super(Singleton, cls).__new__(cls) return cls.instance
餓漢式在創(chuàng)建的時(shí)候就會(huì)生成實(shí)例
# 懶漢式class Singleton(object): __instance = None def __init__(self): if not self.__instance: print(’調(diào)用__init__, 實(shí)例未創(chuàng)建’) else: print(’調(diào)用__init__,實(shí)例已經(jīng)創(chuàng)建過(guò)了:’, __instance) @classmethod def get_instance(cls): # 調(diào)用get_instance類(lèi)方法的時(shí)候才會(huì)生成Singleton實(shí)例 if not cls.__instance: cls.__instance = Singleton() return cls.__instance
以上這篇Python 用__new__方法實(shí)現(xiàn)單例的操作就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持好吧啦網(wǎng)。
相關(guān)文章:
1. HTTP協(xié)議常用的請(qǐng)求頭和響應(yīng)頭響應(yīng)詳解說(shuō)明(學(xué)習(xí))2. HTML DOM setInterval和clearInterval方法案例詳解3. 不要在HTML中濫用div4. React優(yōu)雅的封裝SvgIcon組件示例5. CSS清除浮動(dòng)方法匯總6. react實(shí)現(xiàn)組件狀態(tài)緩存的示例代碼7. HTML5 Canvas繪制圖形從入門(mén)到精通8. Electron調(diào)用外接攝像頭并拍照上傳實(shí)現(xiàn)詳解9. HTML5實(shí)戰(zhàn)與剖析之觸摸事件(touchstart、touchmove和touchend)10. Vue如何使用ElementUI對(duì)表單元素進(jìn)行自定義校驗(yàn)及踩坑
