Python中對象方法的定義很怪異,第一個參數一般都命名為self(相當于其它語言的this),用于傳遞對象本身,而在調用的時候則不必顯式傳遞,系統會自動傳遞。
今天我們介紹的主角是super(), 在類的繼承里面super()非常常用, 它解決了子類調用父類方法的一些問題, 父類多次被調用時只執行一次, 優化了執行邏輯,下面我們就來詳細看一下。
舉一個例子:
class Foo:
def bar(self, message):
print(message)
>>> Foo().bar("Hello, Python.")
Hello, Python.
當存在繼承關系的時候,有時候需要在子類中調用父類的方法,此時最簡單的方法是把對象調用轉換成類調用,需要注意的是這時self參數需要顯式傳遞,例如:
class FooParent:
def bar(self, message):
print(message)
class FooChild(FooParent):
def bar(self, message):
FooParent.bar(self, message)
>>> FooChild().bar("Hello, Python.")
Hello, Python.
這樣做有一些缺點,比如說如果修改了父類名稱,那么在子類中會涉及多處修改,另外,Python是允許多繼承的語言,如上所示的方法在多繼承時就需要重復寫多次,顯得累贅。為了解決這些問題,Python引入了super()機制,例子代碼如下:
class FooParent:
def bar(self, message):
print(message)
class FooChild(FooParent):
def bar(self, message):
super(FooChild, self).bar(message)
>>> FooChild().bar("Hello, Python.")
Hello, Python.
表面上看 super(FooChild, self).bar(message)方法和FooParent.bar(self, message)方法的結果是一致的,實際上這兩種方法的內部處理機制大大不同,當涉及多繼承情況時,就會表現出明顯的差異來,直接給例子:
代碼一:
class A:
def __init__(self):
print("Enter A")
print("Leave A")
class B(A):
def __init__(self):
print("Enter B")
A.__init__(self)
print("Leave B")
class C(A):
def __init__(self):
print("Enter C")
A.__init__(self)
print("Leave C")
class D(A):
def __init__(self):
print("Enter D")
A.__init__(self)
print("Leave D")
class E(B, C, D):
def __init__(self):
print("Enter E")
B.__init__(self)
C.__init__(self)
D.__init__(self)
print("Leave E")
E()
結果:
Enter E
Enter B
Enter A
Leave A
Leave B
Enter C
Enter A
Leave A
Leave C
Enter D
Enter A
Leave A
Leave D
Leave E
執行順序很好理解,唯一需要注意的是公共父類A被執行了多次。
代碼二:
class A:
def __init__(self):
print("Enter A")
print("Leave A")
class B(A):
def __init__(self):
print("Enter B")
super(B, self).__init__()
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, D):
def __init__(self):
print("Enter E")
super(E, self).__init__()
print("Leave E")
E()
結果:
Enter E
Enter B
Enter C
Enter D
Enter A
Leave A
Leave D
Leave C
Leave B
Leave E
在super機制里可以保證公共父類僅被執行一次,至于執行的順序,是按照MRO(Method Resolution Order):方法解析順序 進行的。后續會詳細介紹一下這個MRO機制。
今年第六屆大會PyConChina2016,由PyChina.org發起,CPyUG/TopGeek 等社區協辦,將在2016年9月10日(上海)9月23日(深圳)10月15日(北京)地舉辦的針對Python開發者所舉辦的最盛大和權威的Python相關技術會議,由PyChina社區主辦,致力于推動各類Python相關的技術在互聯網、企業應用等領域的研發和應用。
您可以點擊此處
了解更多詳情,或者掃描下圖二維碼: