一文詳解Python中實(shí)現(xiàn)單例模式的幾種常見(jiàn)方式
Python 中實(shí)現(xiàn)單例模式的幾種常見(jiàn)方式
元類(lèi)(Metaclass):
class SingletonType(type): """ 單例元類(lèi)。用于將普通類(lèi)轉(zhuǎn)換為單例類(lèi)。 """ _instances = {} # 存儲(chǔ)單例實(shí)例的字典 def __call__(cls, *args, **kwargs): """ 重寫(xiě) __call__ 方法。用于創(chuàng)建和返回單例實(shí)例。 """ if cls not in cls._instances: # 如果類(lèi)還沒(méi)有實(shí)例化過(guò) cls._instances[cls] = super().__call__(*args, **kwargs) # 則創(chuàng)建新實(shí)例并存儲(chǔ)在字典中 return cls._instances[cls] # 返回字典中的實(shí)例 class MyClass(metaclass=SingletonType): """ 單例類(lèi)。使用元類(lèi) SingletonType 將其轉(zhuǎn)換為單例類(lèi)。 """ def __init__(self, name): self.name = name def say_hello(self): print(f"Hello, my name is {self.name}.") # 創(chuàng)建 MyClass 的兩個(gè)實(shí)例,應(yīng)該是同一個(gè)對(duì)象 obj1 = MyClass("Alice") obj2 = MyClass("Bob") # 打印兩個(gè)實(shí)例的內(nèi)存地址,應(yīng)該相同 print(hex(id(obj1))) # 輸出:0x7f8d94547a90 print(hex(id(obj2))) # 輸出:0x7f8d94547a90 # 調(diào)用兩個(gè)實(shí)例的方法,輸出應(yīng)該相同 obj1.say_hello() # 輸出:Hello, my name is Alice. obj2.say_hello() # 輸出:Hello, my name is Alice.
在上面的代碼中,我們定義了一個(gè)名為 SingletonType
的元類(lèi),并將其用作 MyClass
的元類(lèi)。在 SingletonType
類(lèi)中,我們維護(hù)了一個(gè) _instances
字典,用于存儲(chǔ)每個(gè)類(lèi)的唯一實(shí)例。在 __call__()
方法中,我們檢查 _instances
字典,如果類(lèi)尚未擁有實(shí)例,則創(chuàng)建一個(gè)新實(shí)例并添加到 _instances
中。最后,我們返回 _instances
中的實(shí)例。
在 MyClass
類(lèi)中,我們定義了一個(gè)帶參數(shù)的構(gòu)造函數(shù),并且使用 metaclass
參數(shù)來(lái)指定 SingletonType
元類(lèi)。由于 MyClass
類(lèi)使用 SingletonType
元類(lèi),因此它具有單例行為。在程序中,我們創(chuàng)建了 MyClass
的兩個(gè)實(shí)例 obj1
和 obj2
,然后打印它們的內(nèi)存地址以驗(yàn)證它們是否是同一個(gè)對(duì)象。最后,我們調(diào)用這兩個(gè)實(shí)例的方法,輸出應(yīng)該相同。
裝飾器(Decorator):
def singleton(cls): """ 單例裝飾器。用于將普通類(lèi)轉(zhuǎn)換為單例類(lèi)。 """ instances = {} # 存儲(chǔ)單例實(shí)例的字典 def get_instance(*args, **kwargs): """ 獲取單例實(shí)例的方法。 """ if cls not in instances: # 如果類(lèi)還沒(méi)有實(shí)例化過(guò) instances[cls] = cls(*args, **kwargs) # 則創(chuàng)建新實(shí)例并存儲(chǔ)在字典中 return instances[cls] # 返回字典中的實(shí)例 return get_instance @singleton class MyClass: """ 單例類(lèi)。使用裝飾器 singleton 將其轉(zhuǎn)換為單例類(lèi)。 """ def __init__(self, name): self.name = name def say_hello(self): print(f"Hello, my name is {self.name}.") # 創(chuàng)建 MyClass 的兩個(gè)實(shí)例,應(yīng)該是同一個(gè)對(duì)象 obj1 = MyClass("Alice") obj2 = MyClass("Bob") # 打印兩個(gè)實(shí)例的內(nèi)存地址,應(yīng)該相同 print(hex(id(obj1))) # 輸出:0x7f8d94547a90 print(hex(id(obj2))) # 輸出:0x7f8d94547
在上面的代碼中,我們定義了一個(gè)名為 singleton
的裝飾器函數(shù)。在 singleton
函數(shù)內(nèi)部,我們創(chuàng)建了一個(gè) instances
字典,用于存儲(chǔ)每個(gè)類(lèi)的唯一實(shí)例。然后,我們定義了一個(gè)名為 get_instance
的內(nèi)部函數(shù),用于獲取單例實(shí)例。在 get_instance
函數(shù)中,我們檢查 instances
字典,如果類(lèi)尚未擁有實(shí)例,則創(chuàng)建一個(gè)新實(shí)例并添加到 instances
中。最后,我們返回字典中的實(shí)例。
在 MyClass
類(lèi)上應(yīng)用 @singleton
裝飾器,以將其轉(zhuǎn)換為單例類(lèi)。由于該裝飾器是針對(duì)類(lèi)進(jìn)行操作的,因此它可以輕松地將任何普通類(lèi)轉(zhuǎn)換為單例類(lèi)。在程序中,我們創(chuàng)建了 MyClass
的兩個(gè)實(shí)例 obj1
和 obj2
,然后打印它們的內(nèi)存地址以驗(yàn)證它們是否是同一個(gè)對(duì)象。最后,我們調(diào)用這兩個(gè)實(shí)例的方法,輸出應(yīng)該相同。
模塊(Module):
# mymodule.py class MyClass: """ 單例類(lèi)。 """ def __init__(self, name): self.name = name def say_hello(self): print(f"Hello, my name is {self.name}.") my_singleton = MyClass("Alice") # 創(chuàng)建單例實(shí)例
# main.py from mymodule import my_singleton # 使用單例實(shí)例 my_singleton.say_hello() # 輸出:Hello, my name is Alice.
在上面的代碼中,我們將 MyClass
類(lèi)定義在一個(gè)獨(dú)立的模塊 mymodule.py
中,并在其中創(chuàng)建了一個(gè)單例實(shí)例 my_singleton
。然后,在另一個(gè)文件 main.py
中,我們從 mymodule
模塊中導(dǎo)入 my_singleton
實(shí)例,并使用它來(lái)調(diào)用 say_hello()
方法。
由于 Python 模塊在首次導(dǎo)入時(shí)會(huì)自動(dòng)執(zhí)行,因此我們可以利用這一特性來(lái)創(chuàng)建單例實(shí)例。在 mymodule.py
模塊中,我們可以確保 my_singleton
只會(huì)被創(chuàng)建一次,并在程序的其他部分中共享它。
new 方法:
class MyClass: """ 單例類(lèi)。 """ _instance = None # 存儲(chǔ)單例實(shí)例的類(lèi)變量 def __new__(cls, *args, **kwargs): """ 重寫(xiě) __new__ 方法。用于創(chuàng)建和返回單例實(shí)例。 """ if cls._instance is None: # 如果類(lèi)還沒(méi)有實(shí)例化過(guò) cls._instance = super().__new__(cls) # 則創(chuàng)建新實(shí)例并存儲(chǔ)在類(lèi)變量中 return cls._instance # 返回類(lèi)變量中的實(shí)例 def __init__(self, name): self.name = name def say_hello(self): print(f"Hello, my name is {self.name}.") # 創(chuàng)建 MyClass 的兩個(gè)實(shí)例,應(yīng)該是同一個(gè)對(duì)象 obj1 = MyClass("Alice") obj2 = MyClass("Bob") # 打印兩個(gè)實(shí)例的內(nèi)存地址,應(yīng)該相同 print(hex(id(obj1))) # 輸出:0x7f8d94547a90 print(hex(id(obj2))) # 輸出:0x7f8d94547a90 # 調(diào)用兩個(gè)實(shí)例的方法,輸出應(yīng)該相同 obj1.say_hello() # 輸出:Hello, my name is Alice. obj2.say_hello() # 輸出:Hello, my name is Alice.
在上面的代碼中,我們將 MyClass
類(lèi)的構(gòu)造函數(shù)改為 __new__()
方法,并使用 _instance
類(lèi)變量來(lái)存儲(chǔ)單例實(shí)例。在 __new__()
方法中,我們檢查 _instance
變量,如果類(lèi)尚未擁有實(shí)例,則創(chuàng)建一個(gè)新實(shí)例并添加到 _instance
中。最后,我們返回 _instance
中的實(shí)例。
在程序中,我們創(chuàng)建了 MyClass
的兩個(gè)實(shí)例 obj1
和 obj2
,然后打印它們的內(nèi)存地址以驗(yàn)證它們是否是同一個(gè)對(duì)象。最后,我們調(diào)用這兩個(gè)實(shí)例的方法,輸出應(yīng)該相同。
無(wú)論使用哪種方法實(shí)現(xiàn)單例模式,都需要注意線程安全和可擴(kuò)展性等方面的問(wèn)題。因此,在實(shí)際開(kāi)發(fā)中,請(qǐng)仔細(xì)考慮自己的需求并選擇合適的實(shí)現(xiàn)方式。
以上就是一文詳解Python中實(shí)現(xiàn)單例模式的幾種常見(jiàn)方式的詳細(xì)內(nèi)容,更多關(guān)于Python 單例模式的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
python利用matplotlib庫(kù)繪制餅圖的方法示例
Python強(qiáng)大的原因之一就在于其開(kāi)源,有很多優(yōu)秀的程序員為其提供了豐富的類(lèi)庫(kù)。Matplotlib就是其中之一,下面這篇文章主要介紹了python如何利用matplotlib庫(kù)繪制餅圖的方法示例,有需要的朋友們可以參考借鑒,下面來(lái)一起看看吧。2016-12-12Python爬蟲(chóng)之網(wǎng)頁(yè)圖片抓取的方法
最近小編一直在學(xué)習(xí)python的東西,今天小編給大家分享基于python寫(xiě)的一個(gè)爬蟲(chóng)程序,能實(shí)現(xiàn)簡(jiǎn)單的網(wǎng)頁(yè)圖片下載,具體實(shí)例代碼大家參考下本文2018-07-07Python數(shù)據(jù)結(jié)構(gòu)之翻轉(zhuǎn)鏈表
這篇文章主要介紹了Python數(shù)據(jù)結(jié)構(gòu)之翻轉(zhuǎn)鏈表的相關(guān)資料,需要的朋友可以參考下2017-02-02Python腳本實(shí)現(xiàn)定時(shí)任務(wù)的最佳方法
我們?cè)谌粘9ぷ髦?常常會(huì)用到需要周期性執(zhí)行的任務(wù),下面這篇文章主要給大家介紹了關(guān)于Python腳本實(shí)現(xiàn)定時(shí)任務(wù)的最佳方法,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-05-05Python實(shí)現(xiàn)統(tǒng)計(jì)給定列表中指定數(shù)字出現(xiàn)次數(shù)的方法
這篇文章主要介紹了Python實(shí)現(xiàn)統(tǒng)計(jì)給定列表中指定數(shù)字出現(xiàn)次數(shù)的方法,涉及Python針對(duì)列表的簡(jiǎn)單遍歷、計(jì)算相關(guān)操作技巧,需要的朋友可以參考下2018-04-04Python編程快速上手——strip()函數(shù)的正則表達(dá)式實(shí)現(xiàn)方法分析
這篇文章主要介紹了Python strip()函數(shù)的正則表達(dá)式實(shí)現(xiàn)方法,結(jié)合實(shí)例形式分析了Python基于正則表達(dá)式實(shí)現(xiàn)strip()函數(shù)的方法,需要的朋友可以參考下2020-02-02