溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Python中單例模式是什么

發(fā)布時間:2020-07-20 14:14:20 來源:億速云 閱讀:118 作者:清晨 欄目:編程語言

這篇文章將為大家詳細講解有關Python中單例模式是什么,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。

簡單而言,單例模式就是保證某個實例在項目的整個生命周期中只存在一個,在項目的任意位置使用,都是同一個實例。

單例模式雖然簡單,但還是有些門道的,而少有人知道這些門道。

邊界情況

Python中實現(xiàn)單例模式的方法很多,我以前最常使用的應該是下面這種寫法。

class Singleton(object):
    
    _instance = None
    def __new__(cls, *args, **kw):
        if cls._instance is None:
            cls._instance = object.__new__(cls, *args, **kw)
        return cls._instance

這種寫法有兩個問題。

1.單例模式對應類實例化時無法傳入?yún)?shù),將上面的代碼擴展成下面形式。

class Singleton(object):
    
    _instance = None
    def __new__(cls, *args, **kw):
        if cls._instance is None:
            cls._instance = object.__new__(cls, *args, **kw)
        return cls._instance

    def __init(self, x, y):
        self.x = x
        self.y = y

s = Singleton(1,2)

此時會拋出TypeError: object.__new__() takes exactly one argument (the type to instantiate)錯誤

2.多個線程實例化Singleton類時,可能會出現(xiàn)創(chuàng)建多個實例的情況,因為很有可能多個線程同時判斷cls._instance is None,從而進入初

始化實例的代碼中。

基于同步鎖實現(xiàn)單例

先考慮上述實現(xiàn)遇到的第二個問題。

既然多線程情況下會出現(xiàn)邊界情況從而參數(shù)多個實例,那么使用同步鎖解決多線程的沖突則可。

import threading

# 同步鎖
def synchronous_lock(func):
    def wrapper(*args, **kwargs):
        with threading.Lock():
            return func(*args, **kwargs)
    return wrapper

class Singleton(object):
    instance = None

    @synchronous_lock
    def __new__(cls, *args, **kwargs):
        if cls.instance is None:
            cls.instance = object.__new__(cls, *args, **kwargs)
        return cls.instance

上述代碼中通過threading.Lock()將單例化方法同步化,這樣在面對多個線程時也不會出現(xiàn)創(chuàng)建多個實例的情況,可以簡單試驗一下。

def worker():
    s = Singleton()
    print(id(s))

def test():
    task = []
    for i in range(10):
        t = threading.Thread(target=worker)
        task.append(t)
    for i in task:
        i.start()
    for i in task:
        i.join()

test()

運行后,打印的單例的id都是相同的。

更優(yōu)的方法

加了同步鎖之后,除了無法傳入?yún)?shù)外,已經(jīng)沒有什么大問題了,但是否有更優(yōu)的解決方法呢?單例模式是否有可以接受參數(shù)的實現(xiàn)方式?

def singleton(cls):
    cls.__new_original__ = cls.__new__

    @functools.wraps(cls.__new__)
    def singleton_new(cls, *args, **kwargs):
        it = cls.__dict__.get('__it__')
        if it is not None:
            return it
        
        cls.__it__ = it = cls.__new_original__(cls, *args, **kwargs)
        it.__init_original__(*args, **kwargs)
        return it

    cls.__new__ = singleton_new
    cls.__init_original__ = cls.__init__
    cls.__init__ = object.__init__
    return cls
    
@singleton
class Foo(object):
    def __new__(cls, *args, **kwargs):
        cls.x = 10
        return object.__new__(cls)

    def __init__(self, x, y):
        assert self.x == 10
        self.x = x
        self.y = y

上述代碼中定義了singleton類裝飾器,裝飾器在預編譯時就會執(zhí)行,利用這個特性,singleton類裝飾器中替換了類原本的__new__與

__init__方法,使用singleton_new方法進行類的實例化,在singleton_new方法中,先判斷類的屬性中是否存在__it__屬性,以此來判斷

是否要創(chuàng)建新的實例,如果要創(chuàng)建,則調(diào)用類原本的__new__方法完成實例化并調(diào)用原本的__init__方法將參數(shù)傳遞給當前類,從而完成單

例模式的目的。

這種方法讓單例類可以接受對應的參數(shù)但面對多線程同時實例化還是可能會出現(xiàn)多個實例,此時加上線程同步鎖則可。

def singleton(cls):
    cls.__new_original__ = cls.__new__
    @functools.wraps(cls.__new__)
    def singleton_new(cls, *args, **kwargs):
        # 同步鎖
        with threading.Lock():
            it = cls.__dict__.get('__it__')
            if it is not None:
                return it
            
            cls.__it__ = it = cls.__new_original__(cls, *args, **kwargs)
            it.__init_original__(*args, **kwargs)
            return it

    cls.__new__ = singleton_new
    cls.__init_original__ = cls.__init__
    cls.__init__ = object.__init__
    return cls

是否加同步鎖的額外考慮

如果一個項目不需要使用線程相關機制,只是在單例化這里使用了線程鎖,這其實不是必要的,它會拖慢項目的運行速度。

閱讀CPython線程模塊相關的源碼,你會發(fā)現(xiàn),Python一開始時并沒有初始化線程相關的環(huán)境,只有當你使用theading庫相關功能時,

才會調(diào)用PyEval_InitThreads方法初始化多線程相關的環(huán)境,代碼片段如下(我省略了很多不相關代碼)。

static PyObject *
thread_PyThread_start_new_thread(PyObject *self, PyObject *fargs)
{
    PyObject *func, *args, *keyw = NULL;
    struct bootstate *boot;
    unsigned long ident;
    
    // 初始化多線程環(huán)境,解釋器默認不初始化,只有用戶使用時,才初始化。
    PyEval_InitThreads(); /* Start the interpreter's thread-awareness */
    // 創(chuàng)建線程
    ident = PyThread_start_new_thread(t_bootstrap, (void*) boot);
    
    // 返回線程id
    return PyLong_FromUnsignedLong(ident);
}

為什么會這樣?

因為多線程環(huán)境會啟動GIL鎖相關的邏輯,這會影響Python程序運行速度。很多簡單的Python程序并不需要使用多線程,此時不需要初始化線程相關的環(huán)境,Python程序在沒有GIL鎖的情況下會運行的更快。

如果你的項目中不會涉及多線程操作,那么就沒有使用有同步鎖來實現(xiàn)單例模式。

互聯(lián)網(wǎng)中有很多Python實現(xiàn)單例模式的文章,你只需要從多線程下是否可以保證單實例以及單例化時是否可以傳入初始參數(shù)兩點來判斷,相應的實現(xiàn)方法則可。

關于Python中單例模式是什么就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,可以學到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權內(nèi)容。

AI