溫馨提示×

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

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

如何在python中實(shí)現(xiàn)單例

發(fā)布時(shí)間:2021-01-06 16:28:03 來源:億速云 閱讀:152 作者:Leah 欄目:開發(fā)技術(shù)

如何在python中實(shí)現(xiàn)單例?相信很多沒有經(jīng)驗(yàn)的人對(duì)此束手無策,為此本文總結(jié)了問題出現(xiàn)的原因和解決方法,通過這篇文章希望你能解決這個(gè)問題。

什么是單例?

確保某一個(gè)類只有一個(gè)實(shí)例,而且自行實(shí)例化并向整個(gè)系統(tǒng)提供這個(gè)實(shí)例,這個(gè)類稱為單例類,單例模式是一種對(duì)象創(chuàng)建型模式。

那么單例模式有什么用途呢?舉個(gè)常見的單例模式例子,我們平時(shí)使用的電腦上都有一個(gè)回收站,在整個(gè)操作系統(tǒng)中,回收站只能有一個(gè)實(shí)例,整個(gè)系統(tǒng)都使用這個(gè)唯一的實(shí)例,而且回收站自行提供自己的實(shí)例,因此回收站是單例模式的應(yīng)用。

裝飾器的方式

這種方式也是工作中經(jīng)常用的一種,用起來也比較方便,代碼實(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è)類需要用單例就通過類似下面的方式實(shí)現(xiàn)即可:

@Singleton
class A(object):

 def __init__(self, x):
  self.x = x

我個(gè)人還是挺喜歡這種方式的

類的方式實(shí)現(xiàn)

這里其實(shí)有一些問題就需要注意了,先看一下可能出現(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è)類好像已經(jīng)實(shí)現(xiàn)了單例,但是這里有一個(gè)潛在的問題,就是如果是多線程的情況,這樣寫就會(huì)有問題了,尤其是在當(dāng)前類的初始化對(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)致你寫的單例就沒起到作用

當(dāng)然自然而然我們會(huì)想起加鎖,通過鎖來控制,所以我們將上面代碼進(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è)問題,就是當(dāng)我們已經(jīng)實(shí)例化過之后每次調(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

看完上述內(nèi)容,你們掌握如何在python中實(shí)現(xiàn)單例的方法了嗎?如果還想學(xué)到更多技能或想了解更多相關(guān)內(nèi)容,歡迎關(guān)注億速云行業(yè)資訊頻道,感謝各位的閱讀!

向AI問一下細(xì)節(jié)

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

AI