在Python類的方法(method)中,要調(diào)用父類的某個方法,在Python 2.2以前,通常的寫法如代碼段1:
代碼段1:
代碼如下:
class B(A):
def __init__(self):
print "enter B"
A.__init__(self)
print "leave B"
>>> b = B()
enter B
enter A
leave A
leave B
這樣做的缺點是,當(dāng)一個子類的父類發(fā)生變化時(如類B的父類由A變?yōu)镃時),必須遍歷整個類定義,把所有的通過非綁定的方法的類名全部替換過來,例如代碼段2,
代碼段2:
代碼如下:
因此,自Python 2.2開始,Python添加了一個關(guān)鍵字super,來解決這個問題。下面是Python 2.3的官方文檔說明:
代碼如下:
Return the superclass of type. If the second argument is omitted the super object
returned is unbound. If the second argument is an object, isinstance(obj, type)
must be true. If the second argument is a type, issubclass(type2, type) must be
true. super() only works for new-style classes.
A typical use for calling a cooperative superclass method is:
class C(B):
def meth(self, arg):
super(C, self).meth(arg)
New in version 2.2.
代碼段3:
代碼如下:
class B(C): # A --> C
def __init__(self):
print "enter B"
super(B, self).__init__()
print "leave B"
嘗試執(zhí)行上面同樣的代碼,結(jié)果一致,但修改的代碼只有一處,把代碼的維護量降到最低,是一個不錯的用法。因此在我們的開發(fā)過程中,super關(guān)鍵字被大量使用,而且一直表現(xiàn)良好。
在我們的印象中,對于super(B, self).__init__()是這樣理解的:super(B, self)首先找到B的父類(就是類A),然后把類B的對象self轉(zhuǎn)換為類A的對象(通過某種方式,一直沒有考究是什么方式,慚愧),然后“被轉(zhuǎn)換”的類A對象調(diào)用自己的__init__函數(shù)。考慮到super中只有指明子類的機制,因此,在多繼承的類定義中,通常我們保留使用類似代碼段1的方法。
有一天某同事設(shè)計了一個相對復(fù)雜的類體系結(jié)構(gòu)(我們先不要管這個類體系設(shè)計得是否合理,僅把這個例子作為一個題目來研究就好),代碼如代碼段4:
代碼段4:
代碼如下:
class B(object):
def __init__(self):
print "enter B"
print "leave B"
class C(A):
def __init__(self):
print "enter C"
super(C, self).__init__()
print "leave C"
class D(A):
def __init__(self):
print "enter D"
super(D, self).__init__()
print "leave D"
class E(B, C):
def __init__(self):
print "enter E"
B.__init__(self)
C.__init__(self)
print "leave E"
class F(E, D):
def __init__(self):
print "enter F"
E.__init__(self)
D.__init__(self)
print "leave F"
>>> f = F()
enter F
enter E
enter B
leave B
enter C
enter D
enter A
leave A
leave D
leave C
leave E
enter D
enter A
leave A
leave D
leave F
二、走進Python的源碼世界
我們嘗試改寫代碼段4中的函數(shù)調(diào)用,但都沒有得到我們想要的結(jié)果,這不得不使我們開始懷疑:我們對super的理解是否出了問題。
我們重新閱讀了Python的官方文檔,正如您所見,官方文檔并沒有詳細的原理說明。到網(wǎng)絡(luò)上去搜索,確實有人發(fā)現(xiàn)了同樣的問題,并在一些論壇中討論,但似乎并沒有實質(zhì)性的解答。既然,沒有前人的足跡,我們只好走進Python的源碼世界,去追溯問題的根源。
我們考查的是Python 2.3的源碼(估計Python 2.4的源碼可能也差不多)。首先,搜索關(guān)鍵字"super"。唯一找到的是bltinmodule.c中的一句:
代碼如下:
在typeobject.c中找到了PySuper_Type的定義:
代碼段5:
代碼如下:
再看superobject的定義:
代碼段6:
代碼如下:
代碼段7:
代碼如下:
查找問題的切入點是為什么在類C中的super調(diào)用會切換到類D的初始化函數(shù)。于是在super_init中添加條件斷點,并跟蹤其后的Python代碼。最終進入到super_getattro函數(shù)——對應(yīng)于super對象訪問名字__init__時的搜索操作。
代碼段8(省略部分無關(guān)代碼,并加入一些注釋):
代碼如下:
starttype = su->obj_type; // 獲得搜索的起點:super對象的obj_type
mro = starttype->tp_mro; // 獲得類的mro
……
for (i = 0; i < n; i++) { // 搜索mro中,定位mro中的type
if ((PyObject *)(su->type) == PyTuple_GET_ITEM(mro, i))
break;
}
i++; // 切換到mro中的下一個類
res = NULL;
for (; i < n; i++) { // 在mro以后的各個命名空間中搜索指定名字
tmp = PyTuple_GET_ITEM(mro, i);
if (PyType_Check(tmp))
dict = ((PyTypeObject *)tmp)->tp_dict;
else if (PyClass_Check(tmp))
dict = ((PyClassObject *)tmp)->cl_dict;
else
continue;
res = PyDict_GetItem(dict, name);
if (res != NULL) {
Py_INCREF(res);
f = res->ob_type->tp_descr_get;
if (f != NULL) {
tmp = f(res, su->obj,
(PyObject *)starttype);
Py_DECREF(res);
res = tmp;
}
return res;
}
}
}
return PyObject_GenericGetAttr(self, name);
}
This field is not inherited; it is calculated fresh by PyType_Ready().
我們把代碼段4改寫為:
代碼段9:
代碼如下:
class B(object):
def __init__(self):
print "enter B"
super(B, self).__init__() # new
print "leave B"
class C(A):
def __init__(self):
print "enter C"
super(C, self).__init__()
print "leave C"
class D(A):
def __init__(self):
print "enter D"
super(D, self).__init__()
print "leave D"
class E(B, C):
def __init__(self):
print "enter E"
super(E, self).__init__() # change
print "leave E"
class F(E, D):
def __init__(self):
print "enter F"
super(F, self).__init__() # change
print "leave F"
>>> f = F()
enter F
enter E
enter B
enter C
enter D
enter A
leave A
leave D
leave C
leave B
leave E
leave F
三、延續(xù)的討論
我們再重新看上面的類體系圖,如果把每一個類看作圖的一個節(jié)點,每一個從子類到父類的直接繼承關(guān)系看作一條有向邊,那么該體系圖將變?yōu)橐粋€有向圖。不能發(fā)現(xiàn)mro的順序正好是該有向圖的一個拓撲排序序列。
從而,我們得到了另一個結(jié)果——Python是如何去處理多繼承。支持多繼承的傳統(tǒng)的面向?qū)ο蟪绦蛘Z言(如C++)是通過虛擬繼承的方式去實現(xiàn)多繼承中父類的構(gòu)造函數(shù)被多次調(diào)用的問題,而Python則通過mro的方式去處理。
但這給我們一個難題:對于提供類體系的編寫者來說,他不知道使用者會怎么使用他的類體系,也就是說,不正確的后續(xù)類,可能會導(dǎo)致原有類體系的錯誤,而且這樣的錯誤非常隱蔽的,也難于發(fā)現(xiàn)。
四、小結(jié)
1. super并不是一個函數(shù),是一個類名,形如super(B, self)事實上調(diào)用了super類的初始化函數(shù),
產(chǎn)生了一個super對象;
2. super類的初始化函數(shù)并沒有做什么特殊的操作,只是簡單記錄了類類型和具體實例;
3. super(B, self).func的調(diào)用并不是用于調(diào)用當(dāng)前類的父類的func函數(shù);
4. Python的多繼承類是通過mro的方式來保證各個父類的函數(shù)被逐一調(diào)用,而且保證每個父類函數(shù)
只調(diào)用一次(如果每個類都使用super);
5. 混用super類和非綁定的函數(shù)是一個危險行為,這可能導(dǎo)致應(yīng)該調(diào)用的父類函數(shù)沒有調(diào)用或者一
個父類函數(shù)被調(diào)用多次。
聲明:本網(wǎng)頁內(nèi)容旨在傳播知識,若有侵權(quán)等問題請及時與本網(wǎng)聯(lián)系,我們將在第一時間刪除處理。TEL:177 7030 7066 E-MAIL:11247931@qq.com