中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

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

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

關于python單例模式的簡介

發布時間:2020-07-02 15:05:47 來源:億速云 閱讀:132 作者:清晨 欄目:編程語言

這篇文章主要介紹關于python單例模式的簡介,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們一定要看完!

 

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

邊界情況

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

1
2
3
4
5
6
7
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.單例模式對應類實例化時無法傳入參數,將上面的代碼擴展成下面形式。

1
2
3
4
5
6
7
8
9
10
11
12
13
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類時,可能會出現創建多個實例的情況,因為很有可能多個線程同時判斷cls._instance is None,從而進入初

始化實例的代碼中。

基于同步鎖實現單例

先考慮上述實現遇到的第二個問題。

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

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()將單例化方法同步化,這樣在面對多個線程時也不會出現創建多個實例的情況,可以簡單試驗一下。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
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都是相同的。

更優的方法

加了同步鎖之后,除了無法傳入參數外,已經沒有什么大問題了,但是否有更優的解決方法呢?單例模式是否有可以接受參數的實現方式?

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
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類裝飾器,裝飾器在預編譯時就會執行,利用這個特性,singleton類裝飾器中替換了類原本的__new__與

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

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

例模式的目的。

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
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線程模塊相關的源碼,你會發現,Python一開始時并沒有初始化線程相關的環境,只有當你使用theading庫相關功能時,

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
static PyObject *
thread_PyThread_start_new_thread(PyObject *self, PyObject *fargs)
{
    PyObject *func, *args, *keyw = NULL;
    struct bootstate *boot;
    unsigned long ident;
     
    // 初始化多線程環境,解釋器默認不初始化,只有用戶使用時,才初始化。
    PyEval_InitThreads(); /* Start the interpreter's thread-awareness */
    // 創建線程
    ident = PyThread_start_new_thread(t_bootstrap, (void*) boot);
     
    // 返回線程id
    return PyLong_FromUnsignedLong(ident);
}

為什么會這樣?

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

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

結尾

1.互聯網中有很多Python實現單例模式的文章,你只需要從多線程下是否可以保證單實例以及單例化時是否可以傳入初始參數兩點來判斷

相應的實現方法則可。

2..光理論是不夠的。這里順便送大家一套2020最新python入門到高級項目實戰視頻教程,可以去小編的Python交流.裙 :七衣衣九七七巴而五(數字的諧音)轉換下可以找到了,還可以跟老司機交流討教!


以上是關于python單例模式的簡介的所有內容,感謝各位的閱讀!希望分享的內容對大家有幫助,更多相關知識,歡迎關注億速云行業資訊頻道!

向AI問一下細節

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

AI

凤凰县| 英山县| 滦平县| 驻马店市| 徐州市| 德令哈市| 金川县| 铅山县| 红桥区| 沐川县| 邳州市| 子长县| 嘉兴市| 小金县| 苏尼特左旗| 建平县| 郴州市| 柞水县| 东至县| 全州县| 米易县| 米林县| 汉中市| 河曲县| 密云县| 高尔夫| 永靖县| 澳门| 崇信县| 思茅市| 天柱县| 修武县| 昔阳县| 屯门区| 丹江口市| 岱山县| 阳曲县| 东阿县| 合水县| 伊春市| 剑阁县|