欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

一文詳解Python中實(shí)現(xiàn)單例模式的幾種常見(jiàn)方式

 更新時(shí)間:2023年03月22日 16:57:14   作者:Dante丶  
這篇文章主要為大家介紹了Python中實(shí)現(xiàn)單例模式的幾種常見(jiàn)方式示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jì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í)例 obj1obj2,然后打印它們的內(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í)例 obj1obj2,然后打印它們的內(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í)例 obj1obj2,然后打印它們的內(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)文章

最新評(píng)論