麻豆黑色丝袜jk制服福利网站-麻豆精品传媒视频观看-麻豆精品传媒一二三区在线视频-麻豆精选传媒4区2021-在线视频99-在线视频a

千鋒教育-做有情懷、有良心、有品質(zhì)的職業(yè)教育機(jī)構(gòu)

手機(jī)站
千鋒教育

千鋒學(xué)習(xí)站 | 隨時(shí)隨地免費(fèi)學(xué)

千鋒教育

掃一掃進(jìn)入千鋒手機(jī)站

領(lǐng)取全套視頻
千鋒教育

關(guān)注千鋒學(xué)習(xí)站小程序
隨時(shí)隨地免費(fèi)學(xué)習(xí)課程

當(dāng)前位置:首頁(yè)  >  技術(shù)干貨  > Python的6種方式實(shí)現(xiàn)單例模式

Python的6種方式實(shí)現(xiàn)單例模式

來(lái)源:千鋒教育
發(fā)布人:xqq
時(shí)間: 2023-11-07 05:09:56 1699304996

單例模式是一個(gè)軟件的設(shè)計(jì)模式,為了保證一個(gè)類,無(wú)論調(diào)用多少次產(chǎn)生的實(shí)例對(duì)象,都是指向同一個(gè)內(nèi)存地址,僅僅只有一個(gè)實(shí)例(只有一個(gè)對(duì)象)。

實(shí)現(xiàn)單例模式的手段有很多種,但總的原則是保證一個(gè)類只要實(shí)例化一個(gè)對(duì)象,下一次再實(shí)例的時(shí)候就直接返回這個(gè)對(duì)象,不再做實(shí)例化的操作。所以這里面的關(guān)鍵一點(diǎn)就是,如何判斷這個(gè)類是否實(shí)例化過(guò)一個(gè)對(duì)象。

這里介紹兩類方式:

一類是通過(guò)模塊導(dǎo)入的方式;

一類是通過(guò)魔法方法判斷的方式;

#基本原理:

-第一類通過(guò)模塊導(dǎo)入的方式,借用了模塊導(dǎo)入時(shí)的底層原理實(shí)現(xiàn)。

-當(dāng)一個(gè)模塊(py文件)被導(dǎo)入時(shí),首先會(huì)執(zhí)行這個(gè)模塊的代碼,然后將這個(gè)模塊的名稱空間加載到內(nèi)存。

-當(dāng)這個(gè)模塊第二次再被導(dǎo)入時(shí),不會(huì)再執(zhí)行該文件,而是直接在內(nèi)存中找。

-于是,如果第一次導(dǎo)入模塊,執(zhí)行文件源代碼時(shí)實(shí)例化了一個(gè)類,那再次導(dǎo)入的時(shí)候,就不會(huì)再實(shí)例化。

-第二類主要是基于類和元類實(shí)現(xiàn),在'對(duì)象'的魔法方法中判斷是否已經(jīng)實(shí)例化過(guò)一個(gè)對(duì)象

-這類方式,根據(jù)實(shí)現(xiàn)的手法不同,又分為不同的方法,如:

-通過(guò)類的綁定方法;通過(guò)元類;通過(guò)類下的__new__;通過(guò)裝飾器(函數(shù)裝飾器,類裝飾器)實(shí)現(xiàn)等。

下面分別介紹這幾種不同的實(shí)現(xiàn)方式,僅供參考實(shí)現(xiàn)思路,不做具體需求。

通過(guò)模塊導(dǎo)入

#cls_singleton.py

classFoo(object):

pass

instance=Foo()

#test.py

importcls_singleton

obj1=cls_singleton.instance

obj2=cls_singleton.instance

print(obj1isobj2)

#原理:模塊第二次導(dǎo)入從內(nèi)存找的機(jī)制

通過(guò)類的綁定方法

classStudent:

_instance=None#記錄實(shí)例化對(duì)象

def__init__(self,name,age):

self.name=name

self.age=age

@classmethod

defget_singleton(cls,*args,**kwargs):

ifnotcls._instance:

cls._instance=cls(*args,**kwargs)

returncls._instance

stu1=Student.get_singleton('jack',18)

stu2=Student.get_singleton('jack',18)

print(stu1isstu2)

print(stu1.__dict__,stu2.__dict__)

#原理:類的綁定方法是第二種實(shí)例化對(duì)象的方式,

#第一次實(shí)例化的對(duì)象保存成類的數(shù)據(jù)屬性_instance,

#第二次再實(shí)例化時(shí),在get_singleton中判斷已經(jīng)有了實(shí)例對(duì)象,直接返回類的數(shù)據(jù)屬性_instance

通過(guò)魔法方法__new__

classStudent:

_instance=None

def__init__(self,name,age):

self.name=name

self.age=age

def__new__(cls,*args,**kwargs):

#ifcls._instance:

#returncls._instance#有實(shí)例則直接返回

#else:

#cls._instance=super().__new__(cls)#沒(méi)有實(shí)例則new一個(gè)并保存

#returncls._instance#這個(gè)返回是給是給init,再實(shí)例化一次,也沒(méi)有關(guān)系

ifnotcls._instance:#這是簡(jiǎn)化的寫(xiě)法,上面注釋的寫(xiě)法更容易提現(xiàn)判斷思路

cls._instance=super().__new__(cls)

returncls._instance

stu1=Student('jack',18)

stu2=Student('jack',18)

print(stu1isstu2)

print(stu1.__dict__,stu2.__dict__)

#原理:和方法2類似,將判斷的實(shí)現(xiàn)方式,從類的綁定方法中轉(zhuǎn)移到類的__new__中

#歸根結(jié)底都是判斷類有沒(méi)有實(shí)例,有則直接返回,無(wú)則實(shí)例化并保存到_instance中。

通過(guò)元類

classMymeta(type):

def__init__(cls,name,bases,dic):

super().__init__(name,bases,dic)

cls._instance=None#將記錄類的實(shí)例對(duì)象的數(shù)據(jù)屬性放在元類中自動(dòng)定義了

def__call__(cls,*args,**kwargs):#此call會(huì)在類被調(diào)用(即實(shí)例化時(shí)觸發(fā))

ifcls._instance:#判斷類有沒(méi)有實(shí)例化對(duì)象

returncls._instance

else:#沒(méi)有實(shí)例化對(duì)象時(shí),控制類造空對(duì)象并初始化

obj=cls.__new__(cls,*args,**kwargs)

obj.__init__(*args,**kwargs)

cls._instance=obj#保存對(duì)象,下一次再實(shí)例化可以直接返回而不用再造對(duì)象

returnobj

classStudent(metaclass=Mymeta):

def__init__(self,name,age):

self.name=name

self.age=age

stu1=Student('jack',18)

stu2=Student('jack',18)

print(stu1isstu2)

print(stu1.__dict__,stu2.__dict__)

#原理:類定義時(shí)會(huì)調(diào)用元類下的__init__,類調(diào)用(實(shí)例化對(duì)象)時(shí)會(huì)觸發(fā)元類下的__call__方法

#類定義時(shí),給類新增一個(gè)空的數(shù)據(jù)屬性,

#第一次實(shí)例化時(shí),實(shí)例化之后就將這個(gè)對(duì)象賦值給類的數(shù)據(jù)屬性;第二次再實(shí)例化時(shí),直接返回類的這個(gè)數(shù)據(jù)屬性

#和方式3的不同之處1:類的這個(gè)數(shù)據(jù)屬性是放在元類中自動(dòng)定義的,而不是在類中顯示的定義的。

#和方式3的不同之處2:類調(diào)用時(shí)觸發(fā)元類__call__方法判斷是否有實(shí)例化對(duì)象,而不是在類的綁定方法中判斷

函數(shù)裝飾器

defsingleton(cls):

_instance_dict={}#采用字典,可以裝飾多個(gè)類,控制多個(gè)類實(shí)現(xiàn)單例模式

definner(*args,**kwargs):

ifclsnotin_instance_dict:

_instance_dict[cls]=cls(*args,**kwargs)

return_instance_dict.get(cls)

returninner

@singleton

classStudent:

def__init__(self,name,age):

self.name=name

self.age=age

#def__new__(cls,*args,**kwargs):#將方法3的這部分代碼搬到了函數(shù)裝飾器中

#ifnotcls._instance:

#cls._instance=super().__new__(cls)

#returncls._instan

stu1=Student('jack',18)

stu2=Student('jack',18)

print(stu1isstu2)

print(stu1.__dict__,stu2.__dict__)

類裝飾器

classSingleTon:

_instance_dict={}

def__init__(self,cls_name):

self.cls_name=cls_name

def__call__(self,*args,**kwargs):

ifself.cls_namenotinSingleTon._instance_dict:

SingleTon._instance_dict[self.cls_name]=self.cls_name(*args,**kwargs)

returnSingleTon._instance_dict.get(self.cls_name)

@SingleTon#這個(gè)語(yǔ)法糖相當(dāng)于Student=SingleTon(Student),即Student是SingleTon的實(shí)例對(duì)象

classStudent:

def__init__(self,name,age):

self.name=name

self.age=age

stu1=Student('jack',18)

stu2=Student('jack',18)

print(stu1isstu2)

print(stu1.__dict__,stu2.__dict__)

#原理:在函數(shù)裝飾器的思路上,將裝飾器封裝成類。

#程序執(zhí)行到與語(yǔ)法糖時(shí),會(huì)實(shí)例化一個(gè)Student對(duì)象,這個(gè)對(duì)象是SingleTon的對(duì)象。

#后面使用的Student本質(zhì)上使用的是SingleTon的對(duì)象。

#所以使用Student('jack',18)來(lái)實(shí)例化對(duì)象,其實(shí)是在調(diào)用SingleTon的對(duì)象,會(huì)觸發(fā)其__call__的執(zhí)行

#所以就在__call__中,判斷Student類有沒(méi)有實(shí)例對(duì)象了。

以上內(nèi)容為大家介紹了Python的6種方式實(shí)現(xiàn)單例模式,希望對(duì)大家有所幫助,如果想要了解更多Python相關(guān)知識(shí),請(qǐng)關(guān)注IT培訓(xùn)機(jī)構(gòu):千鋒教育。http://www.dietsnews.net/

聲明:本站稿件版權(quán)均屬千鋒教育所有,未經(jīng)許可不得擅自轉(zhuǎn)載。
10年以上業(yè)內(nèi)強(qiáng)師集結(jié),手把手帶你蛻變精英
請(qǐng)您保持通訊暢通,專屬學(xué)習(xí)老師24小時(shí)內(nèi)將與您1V1溝通
免費(fèi)領(lǐng)取
今日已有369人領(lǐng)取成功
劉同學(xué) 138****2860 剛剛成功領(lǐng)取
王同學(xué) 131****2015 剛剛成功領(lǐng)取
張同學(xué) 133****4652 剛剛成功領(lǐng)取
李同學(xué) 135****8607 剛剛成功領(lǐng)取
楊同學(xué) 132****5667 剛剛成功領(lǐng)取
岳同學(xué) 134****6652 剛剛成功領(lǐng)取
梁同學(xué) 157****2950 剛剛成功領(lǐng)取
劉同學(xué) 189****1015 剛剛成功領(lǐng)取
張同學(xué) 155****4678 剛剛成功領(lǐng)取
鄒同學(xué) 139****2907 剛剛成功領(lǐng)取
董同學(xué) 138****2867 剛剛成功領(lǐng)取
周同學(xué) 136****3602 剛剛成功領(lǐng)取
相關(guān)推薦HOT
Python和PHP的區(qū)別

從開(kāi)發(fā)的角度來(lái)看,PHP是面向WEB的語(yǔ)言。PHP應(yīng)用程序更像是一組單獨(dú)的腳本,甚至只是一個(gè)單獨(dú)入口。而Python是多用途語(yǔ)言,也可以用于WEB開(kāi)發(fā),...詳情>>

2023-11-07 08:02:45
Python爬蟲(chóng)原理

簡(jiǎn)單來(lái)說(shuō)互聯(lián)網(wǎng)是由一個(gè)個(gè)站點(diǎn)和網(wǎng)絡(luò)設(shè)備組成的大網(wǎng),我們通過(guò)瀏覽器訪問(wèn)站點(diǎn),站點(diǎn)把HTML、JS、CSS代碼返回給瀏覽器,這些代碼經(jīng)過(guò)瀏覽器解析...詳情>>

2023-11-07 07:33:56
使用Cython為Python編寫(xiě)更快的C擴(kuò)展

使用Python很有趣,但有時(shí),用它編寫(xiě)的程序可能很慢。所有的運(yùn)行時(shí)動(dòng)態(tài)調(diào)度會(huì)帶來(lái)很大的代價(jià):有時(shí)它比用C或Rust等系統(tǒng)語(yǔ)言編寫(xiě)的等效代碼慢10...詳情>>

2023-11-07 07:23:08
Python字符串連接

字符串是Python中常用的數(shù)據(jù)類型,在開(kāi)發(fā)過(guò)程中可以對(duì)字符創(chuàng)進(jìn)行截取并與其他字符創(chuàng)進(jìn)行連接。下面小編整理了5種方法完成Python字符創(chuàng)的連接!加...詳情>>

2023-11-07 06:47:08
數(shù)據(jù)科學(xué)中必須了解的Python核心庫(kù)

python有三個(gè)核心數(shù)據(jù)科學(xué)庫(kù),在此基礎(chǔ)上還創(chuàng)建了許多其他的庫(kù)。這三個(gè)核心數(shù)據(jù)科學(xué)庫(kù)分別為:NumpyScipyMatplotlib簡(jiǎn)單起見(jiàn),可以把Numpy視為...詳情>>

2023-11-07 06:39:56
主站蜘蛛池模板: 国产一卡二卡| 四虎成人免费| 美女黄色录像| 美女扒开屁股给男人看无遮挡| 波多野结衣无内裤护士| 乱色| 欧美高清hd| 波多野结衣两部黑人mp4| 国产精品福利一区二区| 久久精品中文字幕一区| 番肉动漫无修在线观看网站| 欧美日韩电影在线| 特级做a爰片毛片免费看| yw在线观看成人免费| 国产日韩美国成人| 一个人看的www高清直播在线观看| 波多野结衣一区二区三区在线观看| 3d动漫精品啪啪一区二区免费| 在公车上被一个接一个| 把胡萝卜立着自己坐上去| 国产欧美va欧美va香蕉在线| 99在线精品视频在线观看| 成人777777| 男人和女人做爽爽视频| 中文字幕久久久久| 欧美a级影院| 亚洲性色高清完整版在线观看| 亚洲成年人专区| 亚洲欧美成人一区二区在线电影| 十七岁高清在线观看| 性做久久久久久免费观看| 精品国产柚木在线观看| 亚洲小视频在线观看| 日本卡一卡二新区| aaa一级黄色片| 国产精品一区二区久久不卡| 黑人干白人| 国产无遮挡又黄又爽高清视| 日韩视频一区| 动漫触手被吸乳羞羞动漫| www.嫩草影院|