一文詳解Python中實(shí)現(xiàn)單例模式的幾種常見方式
Python 中實(shí)現(xiàn)單例模式的幾種常見方式
元類(Metaclass):
class SingletonType(type):
"""
單例元類。用于將普通類轉(zhuǎn)換為單例類。
"""
_instances = {} # 存儲單例實(shí)例的字典
def __call__(cls, *args, **kwargs):
"""
重寫 __call__ 方法。用于創(chuàng)建和返回單例實(shí)例。
"""
if cls not in cls._instances: # 如果類還沒有實(shí)例化過
cls._instances[cls] = super().__call__(*args, **kwargs) # 則創(chuàng)建新實(shí)例并存儲在字典中
return cls._instances[cls] # 返回字典中的實(shí)例
class MyClass(metaclass=SingletonType):
"""
單例類。使用元類 SingletonType 將其轉(zhuǎn)換為單例類。
"""
def __init__(self, name):
self.name = name
def say_hello(self):
print(f"Hello, my name is {self.name}.")
# 創(chuàng)建 MyClass 的兩個實(shí)例,應(yīng)該是同一個對象
obj1 = MyClass("Alice")
obj2 = MyClass("Bob")
# 打印兩個實(shí)例的內(nèi)存地址,應(yīng)該相同
print(hex(id(obj1))) # 輸出:0x7f8d94547a90
print(hex(id(obj2))) # 輸出:0x7f8d94547a90
# 調(diào)用兩個實(shí)例的方法,輸出應(yīng)該相同
obj1.say_hello() # 輸出:Hello, my name is Alice.
obj2.say_hello() # 輸出:Hello, my name is Alice.
在上面的代碼中,我們定義了一個名為 SingletonType 的元類,并將其用作 MyClass 的元類。在 SingletonType 類中,我們維護(hù)了一個 _instances 字典,用于存儲每個類的唯一實(shí)例。在 __call__() 方法中,我們檢查 _instances 字典,如果類尚未擁有實(shí)例,則創(chuàng)建一個新實(shí)例并添加到 _instances 中。最后,我們返回 _instances 中的實(shí)例。
在 MyClass 類中,我們定義了一個帶參數(shù)的構(gòu)造函數(shù),并且使用 metaclass 參數(shù)來指定 SingletonType 元類。由于 MyClass 類使用 SingletonType 元類,因此它具有單例行為。在程序中,我們創(chuàng)建了 MyClass 的兩個實(shí)例 obj1 和 obj2,然后打印它們的內(nèi)存地址以驗(yàn)證它們是否是同一個對象。最后,我們調(diào)用這兩個實(shí)例的方法,輸出應(yīng)該相同。
裝飾器(Decorator):
def singleton(cls):
"""
單例裝飾器。用于將普通類轉(zhuǎn)換為單例類。
"""
instances = {} # 存儲單例實(shí)例的字典
def get_instance(*args, **kwargs):
"""
獲取單例實(shí)例的方法。
"""
if cls not in instances: # 如果類還沒有實(shí)例化過
instances[cls] = cls(*args, **kwargs) # 則創(chuàng)建新實(shí)例并存儲在字典中
return instances[cls] # 返回字典中的實(shí)例
return get_instance
@singleton
class MyClass:
"""
單例類。使用裝飾器 singleton 將其轉(zhuǎn)換為單例類。
"""
def __init__(self, name):
self.name = name
def say_hello(self):
print(f"Hello, my name is {self.name}.")
# 創(chuàng)建 MyClass 的兩個實(shí)例,應(yīng)該是同一個對象
obj1 = MyClass("Alice")
obj2 = MyClass("Bob")
# 打印兩個實(shí)例的內(nèi)存地址,應(yīng)該相同
print(hex(id(obj1))) # 輸出:0x7f8d94547a90
print(hex(id(obj2))) # 輸出:0x7f8d94547
在上面的代碼中,我們定義了一個名為 singleton 的裝飾器函數(shù)。在 singleton 函數(shù)內(nèi)部,我們創(chuàng)建了一個 instances 字典,用于存儲每個類的唯一實(shí)例。然后,我們定義了一個名為 get_instance 的內(nèi)部函數(shù),用于獲取單例實(shí)例。在 get_instance 函數(shù)中,我們檢查 instances 字典,如果類尚未擁有實(shí)例,則創(chuàng)建一個新實(shí)例并添加到 instances 中。最后,我們返回字典中的實(shí)例。
在 MyClass 類上應(yīng)用 @singleton 裝飾器,以將其轉(zhuǎn)換為單例類。由于該裝飾器是針對類進(jìn)行操作的,因此它可以輕松地將任何普通類轉(zhuǎn)換為單例類。在程序中,我們創(chuàng)建了 MyClass 的兩個實(shí)例 obj1 和 obj2,然后打印它們的內(nèi)存地址以驗(yàn)證它們是否是同一個對象。最后,我們調(diào)用這兩個實(shí)例的方法,輸出應(yīng)該相同。
模塊(Module):
# mymodule.py
class MyClass:
"""
單例類。
"""
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 類定義在一個獨(dú)立的模塊 mymodule.py 中,并在其中創(chuàng)建了一個單例實(shí)例 my_singleton。然后,在另一個文件 main.py 中,我們從 mymodule 模塊中導(dǎo)入 my_singleton 實(shí)例,并使用它來調(diào)用 say_hello() 方法。
由于 Python 模塊在首次導(dǎo)入時會自動執(zhí)行,因此我們可以利用這一特性來創(chuàng)建單例實(shí)例。在 mymodule.py 模塊中,我們可以確保 my_singleton 只會被創(chuàng)建一次,并在程序的其他部分中共享它。
new 方法:
class MyClass:
"""
單例類。
"""
_instance = None # 存儲單例實(shí)例的類變量
def __new__(cls, *args, **kwargs):
"""
重寫 __new__ 方法。用于創(chuàng)建和返回單例實(shí)例。
"""
if cls._instance is None: # 如果類還沒有實(shí)例化過
cls._instance = super().__new__(cls) # 則創(chuàng)建新實(shí)例并存儲在類變量中
return cls._instance # 返回類變量中的實(shí)例
def __init__(self, name):
self.name = name
def say_hello(self):
print(f"Hello, my name is {self.name}.")
# 創(chuàng)建 MyClass 的兩個實(shí)例,應(yīng)該是同一個對象
obj1 = MyClass("Alice")
obj2 = MyClass("Bob")
# 打印兩個實(shí)例的內(nèi)存地址,應(yīng)該相同
print(hex(id(obj1))) # 輸出:0x7f8d94547a90
print(hex(id(obj2))) # 輸出:0x7f8d94547a90
# 調(diào)用兩個實(shí)例的方法,輸出應(yīng)該相同
obj1.say_hello() # 輸出:Hello, my name is Alice.
obj2.say_hello() # 輸出:Hello, my name is Alice.
在上面的代碼中,我們將 MyClass 類的構(gòu)造函數(shù)改為 __new__() 方法,并使用 _instance 類變量來存儲單例實(shí)例。在 __new__() 方法中,我們檢查 _instance 變量,如果類尚未擁有實(shí)例,則創(chuàng)建一個新實(shí)例并添加到 _instance 中。最后,我們返回 _instance 中的實(shí)例。
在程序中,我們創(chuàng)建了 MyClass 的兩個實(shí)例 obj1 和 obj2,然后打印它們的內(nèi)存地址以驗(yàn)證它們是否是同一個對象。最后,我們調(diào)用這兩個實(shí)例的方法,輸出應(yīng)該相同。
無論使用哪種方法實(shí)現(xiàn)單例模式,都需要注意線程安全和可擴(kuò)展性等方面的問題。因此,在實(shí)際開發(fā)中,請仔細(xì)考慮自己的需求并選擇合適的實(shí)現(xiàn)方式。
以上就是一文詳解Python中實(shí)現(xiàn)單例模式的幾種常見方式的詳細(xì)內(nèi)容,更多關(guān)于Python 單例模式的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
用Python實(shí)現(xiàn)換行符轉(zhuǎn)換的腳本的教程
這篇文章主要介紹了用Python實(shí)現(xiàn)換行符轉(zhuǎn)換的腳本的教程,代碼非常簡單,包括一個對操作說明的功能的實(shí)現(xiàn),需要的朋友可以參考下2015-04-04
Python3實(shí)現(xiàn)轉(zhuǎn)換Image圖片格式
本篇文章給大家分享了Python3實(shí)現(xiàn)在線轉(zhuǎn)換Image圖片格式的功能以及相關(guān)實(shí)例代碼,有興趣的朋友參考下。2018-06-06
python數(shù)據(jù)結(jié)構(gòu)之二叉樹的統(tǒng)計(jì)與轉(zhuǎn)換實(shí)例
這篇文章主要介紹了python數(shù)據(jù)結(jié)構(gòu)之二叉樹的統(tǒng)計(jì)與轉(zhuǎn)換實(shí)例,例如統(tǒng)計(jì)二叉樹的葉子、分支節(jié)點(diǎn),以及二叉樹的左右兩樹互換等,需要的朋友可以參考下2014-04-04
Python中np.linalg.norm()用法實(shí)例總結(jié)
在線性代數(shù)中一個向量通過矩陣轉(zhuǎn)換成另一個向量時,原有向量的大小就是向量的范數(shù),這個變化過程的大小就是矩陣的范數(shù),下面這篇文章主要給大家介紹了關(guān)于Python中np.linalg.norm()用法的相關(guān)資料,需要的朋友可以參考下2022-07-07
Python 程序報(bào)錯崩潰后如何倒回到崩潰的位置(推薦)
這篇文章主要介紹了Python 程序報(bào)錯崩潰后如何倒回到崩潰的位置,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-06-06
Python中pytest的參數(shù)化實(shí)例解析
這篇文章主要介紹了Python中pytest的參數(shù)化實(shí)例解析,pytest是一個非常成熟的全功能的Python測試框架,主要有簡單靈活,容易上手,支持參數(shù)化等特點(diǎn),需要的朋友可以參考下2023-07-07
python方法如何實(shí)現(xiàn)字符串反轉(zhuǎn)
這篇文章主要介紹了python方法如何實(shí)現(xiàn)字符串反轉(zhuǎn)問題,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-01-01

