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

您的位置:首頁技術文章
文章詳情頁

python 6種方法實現單例模式

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

單例模式是一個軟件的設計模式,為了保證一個類,無論調用多少次產生的實例對象,都是指向同一個內存地址,僅僅只有一個實例(只有一個對象)。

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

這里介紹兩類方式:

一類是通過模塊導入的方式; 一類是通過魔法方法判斷的方式;

# 基本原理:- 第一類通過模塊導入的方式,借用了模塊導入時的底層原理實現。- 當一個模塊(py文件)被導入時,首先會執行這個模塊的代碼,然后將這個模塊的名稱空間加載到內存。- 當這個模塊第二次再被導入時,不會再執行該文件,而是直接在內存中找。- 于是,如果第一次導入模塊,執行文件源代碼時實例化了一個類,那再次導入的時候,就不會再實例化。- 第二類主要是基于類和元類實現,在’對象’的魔法方法中判斷是否已經實例化過一個對象- 這類方式,根據實現的手法不同,又分為不同的方法,如:- 通過類的綁定方法;通過元類;通過類下的__new__;通過裝飾器(函數裝飾器,類裝飾器)實現等。

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

通過模塊導入

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

class Student: _instance = None# 記錄實例化對象 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__)# 原理:類的綁定方法是第二種實例化對象的方式,# 第一次實例化的對象保存成類的數據屬性 _instance,# 第二次再實例化時,在get_singleton中判斷已經有了實例對象,直接返回類的數據屬性 _instance

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

通過魔法方法__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# 有實例則直接返回 # else: # cls._instance = super().__new__(cls)# 沒有實例則new一個并保存 # return cls._instance# 這個返回是給是給init,再實例化一次,也沒有關系 if not cls._instance: # 這是簡化的寫法,上面注釋的寫法更容易提現判斷思路 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類似,將判斷的實現方式,從類的綁定方法中轉移到類的__new__中# 歸根結底都是 判斷類有沒有實例,有則直接返回,無則實例化并保存到_instance中。

補充:這種方式可以近乎完美地實現單例模式,但是依然不夠完美。不完美的地方在于沒有考慮到并發的極端情況下,有可能多個線程同一時刻實例化對象。關于這一點的補充內容在本文的最后一節介紹(!!!進階必會)。

通過元類**

class Mymeta(type): def __init__(cls, name, bases, dic): super().__init__(name, bases, dic) cls._instance = None # 將記錄類的實例對象的數據屬性放在元類中自動定義了 def __call__(cls, *args, **kwargs): # 此call會在類被調用(即實例化時觸發) if cls._instance: # 判斷類有沒有實例化對象 return cls._instance else: # 沒有實例化對象時,控制類造空對象并初始化 obj = cls.__new__(cls, *args, **kwargs) obj.__init__(*args, **kwargs) cls._instance = obj # 保存對象,下一次再實例化可以直接返回而不用再造對象 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__)# 原理:類定義時會調用元類下的__init__,類調用(實例化對象)時會觸發元類下的__call__方法# 類定義時,給類新增一個空的數據屬性,# 第一次實例化時,實例化之后就將這個對象賦值給類的數據屬性;第二次再實例化時,直接返回類的這個數據屬性# 和方式3的不同之處1:類的這個數據屬性是放在元類中自動定義的,而不是在類中顯示的定義的。# 和方式3的不同之處2:類調用時觸發元類__call__方法判斷是否有實例化對象,而不是在類的綁定方法中判斷函數裝飾器

def singleton(cls): _instance_dict = {} # 采用字典,可以裝飾多個類,控制多個類實現單例模式 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的這部分代碼搬到了函數裝飾器中 # 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 # 這個語法糖相當于Student = SingleTon(Student),即Student是SingleTon的實例對象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__)# 原理:在函數裝飾器的思路上,將裝飾器封裝成類。# 程序執行到與語法糖時,會實例化一個Student對象,這個對象是SingleTon的對象。# 后面使用的Student本質上使用的是SingleTon的對象。# 所以使用Student(’jack’, 18)來實例化對象,其實是在調用SingleTon的對象,會觸發其__call__的執行# 所以就在__call__中,判斷Student類有沒有實例對象了。!!!進階必會

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

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

解決的關鍵點,無非就是將具體示例化操作的部分加一把鎖,這樣同時來的多個線程就需要排隊。

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

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

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

以上就是python 6種方法實現單例模式的詳細內容,更多關于python 單例模式的資料請關注好吧啦網其它相關文章!

標簽: Python 編程
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
国产日韩欧美中文在线| 色一区二区三区| 亚洲欧美日本国产专区一区| 日韩精品看片| 国模 一区 二区 三区| 91九色精品| 免费在线成人网| 香蕉久久久久久| 国产精品白丝久久av网站| 欧美日韩18| 精品免费视频| 国产91精品对白在线播放| 亚洲综合精品| 国产欧美日韩视频在线| 福利视频一区| 欧美精品激情| 日本视频中文字幕一区二区三区| 久久这里只有| 国产综合精品| 欧美日韩18| 欧美一区久久久| 先锋影音国产一区| 国产精品视频首页| www.com.cn成人| 久久久水蜜桃av免费网站| 黑丝一区二区三区| 国产亚洲一区| 99视频精品视频高清免费| 蜜臀av性久久久久蜜臀aⅴ流畅 | 欧美日韩国产高清电影| 免费日韩av片| 久久不见久久见国语| 日韩精品一卡| 91麻豆精品激情在线观看最新| 91亚洲一区| 视频一区中文字幕国产| 欧美日韩 国产精品| 91精品韩国| 欧美偷窥清纯综合图区| 成人羞羞在线观看网站| 7777精品| 午夜久久免费观看| 嫩呦国产一区二区三区av| 美女毛片一区二区三区四区| 日韩高清在线不卡| 精品日韩视频| 国产欧美88| 美女黄网久久| 日韩伦理福利| 青青青国产精品| 欧美精品一区二区三区精品| 欧美精品国产| 国产一区成人| 天堂8中文在线最新版在线| 中文字幕亚洲精品乱码| 极品av在线| 国产精品亚洲人成在99www| 国产亚洲在线观看| 日韩精品91| 国产精品国产三级在线观看| 视频在线观看一区| 999精品一区| 精品日韩一区| 日本强好片久久久久久aaa| 一区在线免费| 日韩欧美三级| 久久狠狠亚洲综合| 一二三区精品| 亚洲精品1区| 欧美aa一级| 国产一区二区三区久久| 欧美日韩一区二区三区四区在线观看| 久久xxxx精品视频| 欧美日韩国产免费观看| 久久香蕉国产| 视频小说一区二区| 92国产精品| a日韩av网址| 综合日韩av| 9999国产精品| 国产精品亚洲综合色区韩国| 亚洲青青久久| 日韩精品高清不卡| 亚洲精品无播放器在线播放| 先锋影音国产一区| 亚洲伊人精品酒店| 午夜在线精品偷拍| 伊人久久亚洲| 男女激情视频一区| 蜜臀va亚洲va欧美va天堂| 久久福利精品| 中文不卡在线| 日本成人在线一区| 久久电影一区| 中文字幕日韩高清在线| 天堂va蜜桃一区二区三区| 另类亚洲自拍| 亚洲精品一级二级三级| 亚洲特色特黄| 久久福利精品| 日韩精品视频一区二区三区| 日韩激情一区二区| 国产精品久久久免费| 亚洲天堂一区二区| 久久久久久美女精品| 久久一区二区三区电影| 影音先锋久久| 一区二区91| 国产乱码精品一区二区三区四区| 国产精品多人| 国产高清精品二区| 精品国产一区二| 福利一区二区三区视频在线观看| 国产在线观看www| 999久久久免费精品国产| 亚洲福利精品| 免费国产自线拍一欧美视频| 日韩精品免费视频人成| 免费在线亚洲欧美| 欧美日韩视频网站| 国产精品色网| 国产欧美日韩免费观看| 欧美亚洲日本精品| 国产高清久久| 日韩精品免费观看视频| 美女视频免费精品| 免费福利视频一区二区三区| 国产婷婷精品| 国产精品欧美在线观看| 久久人人99| 日韩黄色免费网站| 国产欧洲在线| 亚洲影院天堂中文av色| 国产一区三区在线播放| 欧美日韩视频一区二区三区| 日韩av不卡在线观看| 久久夜夜操妹子| 日本亚洲视频在线| 日韩精品91| 日韩精品视频中文字幕| yellow在线观看网址| 99热精品在线观看| 麻豆精品国产91久久久久久| 亚州av乱码久久精品蜜桃| 久久精品超碰| 亚洲激情二区| 国产精品毛片一区二区在线看| 亚洲涩涩av| 欧产日产国产精品视频| 亚洲精品美女| 欧美精品资源| 国产伦乱精品| 丝袜脚交一区二区| 水蜜桃久久夜色精品一区| 蜜臀av在线播放一区二区三区| 国产一区二区三区久久久久久久久| 欧美色图一区| 久久av日韩| 中文字幕一区二区三区日韩精品| 国产高清不卡| 九一精品国产| 日本久久综合| 国产欧美日韩影院| 亚洲欧美日韩国产一区二区| 毛片在线网站| 国产极品久久久久久久久波多结野| 黑丝一区二区| 桃色av一区二区| 国产精品久久久久毛片大屁完整版| 先锋影音国产一区| 久久精品高清| 老司机精品视频网| 青青草国产成人99久久| 视频精品一区二区| 日本精品影院| 国内精品美女在线观看| 日本久久二区| 蜜桃久久av一区| 美女久久久久| 超碰超碰人人人人精品| 国产精品免费大片| 日本v片在线高清不卡在线观看| 亚州av乱码久久精品蜜桃| 国产精品久久久久久久久久10秀| 国产色噜噜噜91在线精品| 中文字幕一区二区三区四区久久| 99久久婷婷| 在线观看精品| 国产色播av在线| 国产成人精品一区二区免费看京| 日韩激情一二三区| 午夜日韩影院| 午夜久久av| 日韩精品导航| 天堂va在线高清一区| 视频一区欧美精品| 国产精品试看| 蜜臀av性久久久久蜜臀aⅴ流畅| 中文亚洲欧美| 国产午夜精品一区二区三区欧美 |