python 單例模式

2022-07-03 05:48:10 字數 3150 閱讀 8412

研究了一下python的單利模式,簡單地記錄一下.

1.類和例項的建立過程

從建立類到初始化乙個物件,簡單地可以分為四個步驟:

(1)通過元類建立乙個類,呼叫__new__完成類的建立(階段a)

(2)初始化乙個類,呼叫__init__,初始化乙個類(階段b)

(3)通過類建立乙個物件,呼叫__new__(階段d)

(4)初始化乙個物件,呼叫__init__(階段e)

圖中階段a,階段b,階段c是由元類完成的,在檔案載入的時候,自動呼叫__new__和__init__,階段e不是必須的.

階段d,階段e是建立乙個例項的過程,手動呼叫__new__和__init__.一般情況下,建立乙個類只用例項化乙個類,即呼叫__init__,而__new__的執行被隱藏了,主要是繼承了父類的__new__方法

說完了類和例項的建立過程之後,考慮一下單例模式應該在這四個階段中哪個階段控制比較合適?

(1)階段a:

建立乙個類的過程,也就是說類的建立並未完成.單例是建立例項上,例項的建立又是建立在類上面的,這個階段類還沒有建立,也就不太適合

(2)階段b:

例項化乙個類,這個階段類也建立完成了,似乎有機會控制例項的個數

(3)階段d:

類的建立和例項化完成了,開始建立乙個例項,但還未完成,也可以控制例項的個數

(4)階段e:

這個階段,物件已經建立了,每一次例項化乙個類時,物件已經建立了,控制不了例項的個數.

簡單說,階段a (建立類的過程)類沒有建立,不宜控制例項數量.階段e(類例項化的過程)例項已經建立,控制不了例項數量.在階段b(例項化乙個類)和階段d(建立乙個例項)比較容易控制例項的數量,也就最利於實現單例模式.

2.實現單例模式

(1)階段b(例項化乙個類)

主要是通過建立元類進行控制:

#

b階段class

singleton_c(type):

def__init__(self, *args, **kwargs):

self.

__instance =none

super(singleton_c, self).

__init__(*args, **kwargs)

def__call__(self, *args, **kwargs):

if self.__instance

isnone:

self.

__instance = super(singleton_c, self).__call__(*args, **kwargs)

return self.__instance

else

:

return self.__instance

class

c:

__metaclass__ =singleton_c

def__init__

(self):

print

'i am c

'a =c()

b =c()

print a is b #

true

(2)階段d(建立例項的過程)

#

d階段class

singleton_d(object):

'''例項儲存在sigleton_b類的字典中

'''def

__new__(cls, *args, **kwargs):

ifnot hasattr(cls, '

_instance'):

cls._instance = super(singleton_b, cls).__new__(cls, *args, **kwargs)

return

cls._instance

#先繼承其他類時, 會導致單例模式失效

class

d(singleton_d):

def__init__

(self):

print

'i am d

'a =d()

b =d()

print a is b #

true

(3)階段d(建立例項的過程)   --- 乙個偽單例模式

每一次實現的物件不同,但共用變數

#

d階段class

singleton_d1(object):

_store ={}

def__new__(cls, *args, **kwargs):

obj = super(singleton_b1, cls).__new__(cls, *args, **kwargs)

print

cls, obj, id(cls._store), id(obj._store), id(singleton_b1._store)

obj.

__dict__ =cls._store

return

obj#

先繼承其他類時, 會導致單例模式失效

class

f(singleton_d1):

def__init__

(self, name):

self.name =name

f1 = f('python')

f2 = f('singleton')

print f1 is f2 #

false

(4)裝飾器

def

singleton(cls):

obj ={}

if cls not

inobj:

obj[cls] = cls(*args, **kwargs)

return

obj[cls]

return

@singleton

class

foo(object):

pass

f1 =foo()

f2 =foo()

print f1 is f2 #

true

python單例模式繼承 python單例模式

我們可以使用 new 這個特殊方法。該方法可以建立乙個其所在類的子類的物件。更可喜的是,我們的內建 object 基類實現了 new 方法,所以我們只需讓 sing 類繼承 object 類,就可以利用 object 的 new 方法來建立 sing 物件了。classsing object def...

單例模式 python

單例模式 保證乙個類僅有乙個例項,並提供乙個訪問它的全域性訪問點。實現 某個類只有乙個例項 途徑 1 讓乙個全域性變數使得乙個物件被訪問,但是它不能防止外部例項化多個物件。2 讓類自身負責儲存它的唯一例項。這個類可以保證沒有其他例項可以被建立。即單例模式。多執行緒時的單例模式 加鎖 雙重鎖定。餓漢式...

python單例模式

new 在 init 之前被呼叫,用於生成例項物件。利用這個方法和類的屬性的特點可以實現設計模式的單例模式。單例模式是指建立唯一物件,單例模式設計的類只能例項 例項化1個物件。class singleton object instance none def init self pass def ne...