python中單例常用的幾種實(shí)現(xiàn)方法總結(jié)
前言
最近這兩天在看自己之前寫(xiě)的代碼,所以正好把用過(guò)的東西整理一下,單例模式,在日常的代碼工作中也是經(jīng)常被用到,
所以這里把之前用過(guò)的不同方式實(shí)現(xiàn)的單例方式整理一下
什么是單例?
確保某一個(gè)類(lèi)只有一個(gè)實(shí)例,而且自行實(shí)例化并向整個(gè)系統(tǒng)提供這個(gè)實(shí)例,這個(gè)類(lèi)稱(chēng)為單例類(lèi),單例模式是一種對(duì)象創(chuàng)建型模式。
那么單例模式有什么用途呢?舉個(gè)常見(jiàn)的單例模式例子,我們平時(shí)使用的電腦上都有一個(gè)回收站,在整個(gè)操作系統(tǒng)中,回收站只能有一個(gè)實(shí)例,整個(gè)系統(tǒng)都使用這個(gè)唯一的實(shí)例,而且回收站自行提供自己的實(shí)例,因此回收站是單例模式的應(yīng)用。
裝飾器的方式
這種方式也是工作中經(jīng)常用的一種,用起來(lái)也比較方便,代碼實(shí)現(xiàn)如下
def Singleton(cls):
_instance = {}
def _singleton(*args, **kwargs):
if cls not in _instance:
_instance[cls] = cls(*args, **kwargs)
return _instance[cls]
return _singleton
如果我們工作的一個(gè)類(lèi)需要用單例就通過(guò)類(lèi)似下面的方式實(shí)現(xiàn)即可:
@Singleton class A(object): def __init__(self, x): self.x = x
我個(gè)人還是挺喜歡這種方式的
類(lèi)的方式實(shí)現(xiàn)
這里其實(shí)有一些問(wèn)題就需要注意了,先看一下可能出現(xiàn)的錯(cuò)誤代碼
class Member(object): @classmethod def instance(cls, *args, **kwargs): if not hasattr(Member, "_instance"): Member._instance = Member(*args, **kwargs) return Member._instance
乍一看這個(gè)類(lèi)好像已經(jīng)實(shí)現(xiàn)了單例,但是這里有一個(gè)潛在的問(wèn)題,就是如果是多線程的情況,這樣寫(xiě)就會(huì)有問(wèn)題了,尤其是在當(dāng)前類(lèi)的初始化對(duì)象里有一些耗時(shí)操作時(shí)候
例如下面代碼:
#! /usr/bin/env python3 # .-*- coding:utf-8 .-*- import time import threading import random class Member(object): def __init__(self): time.sleep(random.randint(1,3)) @classmethod def instance(cls, *args, **kwargs): if not hasattr(Member, "_instance"): Member._instance = Member(*args, **kwargs) return Member._instance def task(arg): obj = Member.instance() print(obj) for i in range(5): t = threading.Thread(target=task, args=[i,]) t.start()
這段代碼的執(zhí)行結(jié)果會(huì)出現(xiàn)實(shí)例化了多個(gè)對(duì)象,導(dǎo)致你寫(xiě)的單例就沒(méi)起到作用
當(dāng)然自然而然我們會(huì)想起加鎖,通過(guò)鎖來(lái)控制,所以我們將上面代碼進(jìn)行更改:
#! /usr/bin/env python3
# .-*- coding:utf-8 .-*-
import time
import threading
import random
class Member(object):
_instance_lock = threading.Lock()
def __init__(self):
i = random.randint(1, 3)
print(i)
time.sleep(i)
@classmethod
def instance(cls, *args, **kwargs):
with Member._instance_lock:
if not hasattr(Member, "_instance"):
Member._instance = Member(*args, **kwargs)
return Member._instance
def task():
obj = Member.instance()
print(obj)
for i in range(5):
threading.Thread(target=task,).start()
但是上面的代碼還有一個(gè)問(wèn)題,就是當(dāng)我們已經(jīng)實(shí)例化過(guò)之后每次調(diào)用instance都會(huì)去請(qǐng)求鎖,所以這點(diǎn)并不好,所以我們將這部分代碼再次更改:
@classmethod
def instance(cls, *args, **kwargs):
if not hasattr(Member, "_instance"):
with Member._instance_lock:
if not hasattr(Member, "_instance"):
Member._instance = Member(*args, **kwargs)
return Member._instance
這樣就很好的實(shí)現(xiàn)一個(gè)可以多線程使用的單例
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
C語(yǔ)言中printf()函數(shù)的全面介紹及用法(簡(jiǎn)單易懂)
在C語(yǔ)言中,printf()是常用的輸出函數(shù),包含在頭文件中,它使用格式控制字符串,其中包括格式字符、轉(zhuǎn)義字符和普通字符,格式字符以"%"開(kāi)頭,文中將用法介紹的非常詳細(xì),需要的朋友可以參考下2024-09-09
淺析Python 抽象工廠模式的優(yōu)缺點(diǎn)
這篇文章主要介紹了Python 抽象工廠模式的優(yōu)缺點(diǎn),文中示例代碼非常詳細(xì),幫助大家更好的理解和學(xué)習(xí),感興趣的朋友可以了解下2020-07-07
Python實(shí)現(xiàn)樸素貝葉斯的學(xué)習(xí)與分類(lèi)過(guò)程解析
這篇文章主要介紹了Python實(shí)現(xiàn)樸素貝葉斯的學(xué)習(xí)與分類(lèi)過(guò)程解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-08-08
pycharm運(yùn)行程序時(shí)出現(xiàn)Run‘python tests for XXX.py‘問(wèn)題及
這篇文章主要介紹了pycharm運(yùn)行程序時(shí)出現(xiàn)Run ‘python tests for XXX.py‘問(wèn)題及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-08-08
淺談Python 敏感詞過(guò)濾的實(shí)現(xiàn)
這篇文章主要介紹了淺談Python 敏感詞過(guò)濾的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-08-08
Python讀取環(huán)境變量的方法和自定義類(lèi)分享
這篇文章主要介紹了Python讀取環(huán)境變量的方法和自定義類(lèi)分享,本文直接給出代碼實(shí)例,需要的朋友可以參考下2014-11-11
Pandas數(shù)值排序 sort_values()的使用
本文主要介紹了Pandas數(shù)值排序 sort_values()的使用,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-07-07
Python實(shí)現(xiàn)語(yǔ)音轉(zhuǎn)文本的兩種方法
這篇文章主要給大家介紹了關(guān)于Python實(shí)現(xiàn)語(yǔ)音轉(zhuǎn)文本的兩種方法,Python提供了許多工具和庫(kù)來(lái)進(jìn)行這些任務(wù),本文通過(guò)實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-06-06

