日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術(shù)文章
文章詳情頁

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

瀏覽:28日期:2022-07-02 10:16:25

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

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

這里介紹兩類方式:

一類是通過模塊導(dǎo)入的方式; 一類是通過魔法方法判斷的方式;

# 基本原理:- 第一類通過模塊導(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),在’對象’的魔法方法中判斷是否已經(jīng)實(shí)例化過一個(gè)對象- 這類方式,根據(jù)實(shí)現(xiàn)的手法不同,又分為不同的方法,如:- 通過類的綁定方法;通過元類;通過類下的__new__;通過裝飾器(函數(shù)裝飾器,類裝飾器)實(shí)現(xiàn)等。

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

通過模塊導(dǎo)入

# cls_singleton.pyclass Foo(object): passinstance = Foo()# test.pyimport cls_singletonobj1 = cls_singleton.instanceobj2 = cls_singleton.instanceprint(obj1 is obj2)# 原理:模塊第二次導(dǎo)入從內(nèi)存找的機(jī)制通過類的綁定方法

class Student: _instance = None# 記錄實(shí)例化對象 def __init__(self, name, age): self.name = name self.age = age @classmethod def get_singleton(cls, *args, **kwargs): if not cls._instance: cls._instance = cls(*args, **kwargs) return cls._instancestu1 = Student.get_singleton(’jack’, 18)stu2 = Student.get_singleton(’jack’, 18)print(stu1 is stu2)print(stu1.__dict__, stu2.__dict__)# 原理:類的綁定方法是第二種實(shí)例化對象的方式,# 第一次實(shí)例化的對象保存成類的數(shù)據(jù)屬性 _instance,# 第二次再實(shí)例化時(shí),在get_singleton中判斷已經(jīng)有了實(shí)例對象,直接返回類的數(shù)據(jù)屬性 _instance

補(bǔ)充:這種方式實(shí)現(xiàn)的單例模式有一個(gè)明顯的bug;bug的根源在于如果用戶不通過綁定類的方法實(shí)例化對象,而是直接通過類名加括號(hào)實(shí)例化對象,那這樣不再是單利模式了。

通過魔法方法__new__

class Student: _instance = None def __init__(self, name, age): self.name = name self.age = age def __new__(cls, *args, **kwargs): # if cls._instance: # return cls._instance# 有實(shí)例則直接返回 # else: # cls._instance = super().__new__(cls)# 沒有實(shí)例則new一個(gè)并保存 # return cls._instance# 這個(gè)返回是給是給init,再實(shí)例化一次,也沒有關(guān)系 if not cls._instance: # 這是簡化的寫法,上面注釋的寫法更容易提現(xiàn)判斷思路 cls._instance = super().__new__(cls) return cls._instancestu1 = Student(’jack’, 18)stu2 = Student(’jack’, 18)print(stu1 is stu2)print(stu1.__dict__, stu2.__dict__)# 原理:和方法2類似,將判斷的實(shí)現(xiàn)方式,從類的綁定方法中轉(zhuǎn)移到類的__new__中# 歸根結(jié)底都是 判斷類有沒有實(shí)例,有則直接返回,無則實(shí)例化并保存到_instance中。

補(bǔ)充:這種方式可以近乎完美地實(shí)現(xiàn)單例模式,但是依然不夠完美。不完美的地方在于沒有考慮到并發(fā)的極端情況下,有可能多個(gè)線程同一時(shí)刻實(shí)例化對象。關(guān)于這一點(diǎn)的補(bǔ)充內(nèi)容在本文的最后一節(jié)介紹(!!!進(jìn)階必會(huì))。

通過元類**

class Mymeta(type): def __init__(cls, name, bases, dic): super().__init__(name, bases, dic) cls._instance = None # 將記錄類的實(shí)例對象的數(shù)據(jù)屬性放在元類中自動(dòng)定義了 def __call__(cls, *args, **kwargs): # 此call會(huì)在類被調(diào)用(即實(shí)例化時(shí)觸發(fā)) if cls._instance: # 判斷類有沒有實(shí)例化對象 return cls._instance else: # 沒有實(shí)例化對象時(shí),控制類造空對象并初始化 obj = cls.__new__(cls, *args, **kwargs) obj.__init__(*args, **kwargs) cls._instance = obj # 保存對象,下一次再實(shí)例化可以直接返回而不用再造對象 return objclass Student(metaclass=Mymeta): def __init__(self, name, age): self.name = name self.age = agestu1 = Student(’jack’, 18)stu2 = Student(’jack’, 18)print(stu1 is stu2)print(stu1.__dict__, stu2.__dict__)# 原理:類定義時(shí)會(huì)調(diào)用元類下的__init__,類調(diào)用(實(shí)例化對象)時(shí)會(huì)觸發(fā)元類下的__call__方法# 類定義時(shí),給類新增一個(gè)空的數(shù)據(jù)屬性,# 第一次實(shí)例化時(shí),實(shí)例化之后就將這個(gè)對象賦值給類的數(shù)據(jù)屬性;第二次再實(shí)例化時(shí),直接返回類的這個(gè)數(shù)據(jù)屬性# 和方式3的不同之處1:類的這個(gè)數(shù)據(jù)屬性是放在元類中自動(dòng)定義的,而不是在類中顯示的定義的。# 和方式3的不同之處2:類調(diào)用時(shí)觸發(fā)元類__call__方法判斷是否有實(shí)例化對象,而不是在類的綁定方法中判斷函數(shù)裝飾器

def singleton(cls): _instance_dict = {} # 采用字典,可以裝飾多個(gè)類,控制多個(gè)類實(shí)現(xiàn)單例模式 def inner(*args, **kwargs): if cls not in _instance_dict: _instance_dict[cls] = cls(*args, **kwargs) return _instance_dict.get(cls) return inner@singletonclass Student: def __init__(self, name, age): self.name = name self.age = age # def __new__(cls, *args, **kwargs): # 將方法3的這部分代碼搬到了函數(shù)裝飾器中 # if not cls._instance: # cls._instance = super().__new__(cls) # return cls._instan stu1 = Student(’jack’, 18)stu2 = Student(’jack’, 18)print(stu1 is stu2)print(stu1.__dict__, stu2.__dict__)類裝飾器

class SingleTon: _instance_dict = {} def __init__(self, cls_name): self.cls_name = cls_name def __call__(self, *args, **kwargs): if self.cls_name not in SingleTon._instance_dict: SingleTon._instance_dict[self.cls_name] = self.cls_name(*args, **kwargs) return SingleTon._instance_dict.get(self.cls_name)@SingleTon # 這個(gè)語法糖相當(dāng)于Student = SingleTon(Student),即Student是SingleTon的實(shí)例對象class Student: def __init__(self, name, age): self.name = name self.age = agestu1 = Student(’jack’, 18)stu2 = Student(’jack’, 18)print(stu1 is stu2)print(stu1.__dict__, stu2.__dict__)# 原理:在函數(shù)裝飾器的思路上,將裝飾器封裝成類。# 程序執(zhí)行到與語法糖時(shí),會(huì)實(shí)例化一個(gè)Student對象,這個(gè)對象是SingleTon的對象。# 后面使用的Student本質(zhì)上使用的是SingleTon的對象。# 所以使用Student(’jack’, 18)來實(shí)例化對象,其實(shí)是在調(diào)用SingleTon的對象,會(huì)觸發(fā)其__call__的執(zhí)行# 所以就在__call__中,判斷Student類有沒有實(shí)例對象了。!!!進(jìn)階必會(huì)

本部分主要是補(bǔ)充介紹多線程并發(fā)情況下,多線程高并發(fā)時(shí),如果同時(shí)有多個(gè)線程同一時(shí)刻(極端條件下)事例化對象,那么就會(huì)出現(xiàn)多個(gè)對象,這就不再是單例模式了。

解決這個(gè)多線程并發(fā)帶來的競爭問題,第一個(gè)想到的是加互斥鎖,于是我們就用互斥鎖的原理來解決這個(gè)問題。

解決的關(guān)鍵點(diǎn),無非就是將具體示例化操作的部分加一把鎖,這樣同時(shí)來的多個(gè)線程就需要排隊(duì)。

這樣一來只有第一個(gè)搶到鎖的線程實(shí)例化一個(gè)對象并保存在_instance中,同一時(shí)刻搶鎖的其他線程再搶到鎖后,不會(huì)進(jìn)入這個(gè)判斷if not cls._instance,直接把保存在_instance的對象返回了。這樣就實(shí)現(xiàn)了多線程下的單例模式。

此時(shí)還有一個(gè)問題需要解決,后面所有再事例對象時(shí)都需要再次搶鎖,這會(huì)大大降低執(zhí)行效率。解決這個(gè)問題也很簡單,直接在搶鎖前,判斷下是否有單例對象了,如果有就不再往下?lián)屾i了(代碼第11行判斷存在的意義)。

import threadingclass Student: _instance = None# 保存單例對象 _lock = threading.RLock() # 鎖 def __new__(cls, *args, **kwargs):if cls._instance:# 如果已經(jīng)有單例了就不再去搶鎖,避免IO等待 return cls._instancewith cls._lock:# 使用with語法,方便搶鎖釋放鎖 if not cls._instance:cls._instance = super().__new__(cls, *args, **kwargs) return cls._instance

以上就是python 6種方法實(shí)現(xiàn)單例模式的詳細(xì)內(nèi)容,更多關(guān)于python 單例模式的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: Python 編程
相關(guān)文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
精品资源在线| 国产视频一区在线观看一区免费| 老鸭窝亚洲一区二区三区| 99视频精品| 蜜臀va亚洲va欧美va天堂| 中文一区一区三区免费在线观 | 夜夜精品视频| 亚洲一区二区三区四区电影| 日韩国产成人精品| 毛片不卡一区二区| 国产极品一区| 日韩电影免费网址| 国产亚洲精品v| 国产日韩欧美中文在线| 中文字幕一区久| 免费在线观看不卡| 国产精品传媒麻豆hd| 四虎4545www国产精品 | 午夜亚洲福利| 国产精品亚洲综合色区韩国| 色综合五月天| 亚洲一区二区三区四区五区午夜| 日本不卡一二三区黄网| 高清av不卡| 综合干狼人综合首页| 精品视频高潮| 午夜一级在线看亚洲| 日韩精品1区2区3区| 国产aⅴ精品一区二区四区| 亚洲激情黄色| 免费精品一区| 国产精品婷婷| 久久69成人| 影音先锋久久精品| 亚洲一区资源| 91嫩草精品| 亚洲天堂久久| 国产精品一区二区精品视频观看| 欧美亚洲国产精品久久| 国产精品欧美一区二区三区不卡| 欧美在线资源| 高清一区二区三区av| 天堂精品久久久久| 久久亚洲成人| 精品一区不卡| 日本91福利区| 国产综合欧美| 久久精品人人| 日韩国产在线不卡视频| 亚洲成人精选| 国产高清不卡| 国产精品片aa在线观看| 蜜臀av一区二区三区| 视频小说一区二区| 精品国产亚洲一区二区三区在线| 一二三区精品| 欧美日韩在线网站| 国产一区丝袜| 国产日韩欧美一区二区三区 | 日韩欧美在线中字| 欧美亚洲国产日韩| 伊人国产精品| 国产亚洲永久域名| 在线观看精品| 福利精品在线| 久久伊人久久| 国产香蕉精品| 深夜福利一区| 久久夜色精品| 久久精品国产大片免费观看| 精品久久99| 麻豆一区二区三| 国产精品chinese| 欧美久久久网站| 日韩av中文在线观看| 亚洲另类av| 一本综合精品| 综合亚洲色图| 亚洲色图国产| 日韩中文字幕不卡| 爽爽淫人综合网网站 | 日韩欧美一区二区三区在线观看| 国产精品亚洲片在线播放| 亚洲久久一区| 一区二区三区网站| 香蕉久久一区| 日本精品一区二区三区在线观看视频| 亚洲一区二区三区久久久| 久久国产精品久久w女人spa| 国内精品福利| 亚洲中午字幕| 亚洲一区二区三区久久久| 免费一级片91| 涩涩涩久久久成人精品| 欧美亚洲色图校园春色| 国产亚洲一区二区三区不卡 | 久久99性xxx老妇胖精品| 亚洲精品欧美| 国产日韩中文在线中文字幕| 另类综合日韩欧美亚洲| 精品欧美日韩精品| 日韩深夜视频| 午夜视频精品| 热久久免费视频| 五月激激激综合网色播| 日韩激情精品| 精品一区二区三区在线观看视频| 日韩av在线播放网址| 日韩精品免费一区二区三区| 视频一区中文| 日韩在线视频一区二区三区| 国产欧美综合一区二区三区| 国产精品探花在线观看| 成人在线免费观看网站| 亚洲不卡av不卡一区二区| 午夜久久影院| 日韩激情av在线| 风间由美中文字幕在线看视频国产欧美| 激情国产在线| 99国产精品久久久久久久成人热| 亚洲精品一级| 国产在视频一区二区三区吞精| 精品丝袜在线| 亚洲麻豆一区| 91亚洲一区| 六月天综合网| 卡一卡二国产精品| 国产99亚洲| 日本成人在线视频网站| 亚洲黄色中文字幕| 伊人久久亚洲| 97人人精品| 亚洲久草在线| 97在线精品| 免费视频最近日韩| 手机在线电影一区| 在线日韩成人| 日韩免费久久| 奇米亚洲欧美| 久久久夜夜夜| 国产麻豆精品| 亚洲深夜av| 国产精品玖玖玖在线资源| 亚洲高清毛片| 免费在线观看一区| 国产亚洲午夜| 国产一区二区三区黄网站| 天堂成人免费av电影一区| 精品久久久中文字幕| 亚洲欧美视频| 美女一区网站| 国产午夜精品一区在线观看| 欧美亚洲精品在线| 久久免费福利| 在线国产精品一区| 久久久久国产一区二区| 久久国产婷婷国产香蕉| 欧美精品一二| 黄色精品视频| 日韩精品欧美成人高清一区二区| 久久激情一区| 国产精品v日韩精品v欧美精品网站| 亚洲电影在线一区二区三区| 韩国一区二区三区视频| 日韩综合一区二区三区| 激情五月综合| www.九色在线| 国产高清日韩| 91嫩草精品| 蜜臀av一区二区在线免费观看| 蜜臀久久精品| 久久只有精品| 日韩精品久久久久久| 一区免费在线| 日韩精品首页| 狠狠躁少妇一区二区三区| 国产美女亚洲精品7777| 久久高清免费观看| 欧美日韩国产综合网| 蜜桃成人精品| 三上亚洲一区二区| 精品网站999| 久久中文字幕一区二区三区| 日本视频中文字幕一区二区三区| 日韩影院在线观看| 好看的亚洲午夜视频在线| 99精品视频在线| 捆绑调教日本一区二区三区| 美女久久久久久| 国产精品主播| 国产欧美综合一区二区三区| 日韩精品视频一区二区三区| 中文视频一区| 亚洲网址在线观看| 最新亚洲国产| 日韩精品视频在线看| 亚洲人成网77777色在线播放 | 亚洲精品欧美| 亚洲免费福利一区| 最新国产精品久久久|