新式類和經典類得區別:
###################################################
class ClassicClass():
pass
class NewStyleClass(object):
pass
x1 = ClassicClass()
x2 = NewStyleClass()
print x1.__class__, type(x1)
print x2.__class__, type(x2)
輸出結果:
__main__.ClassicClass <type 'instance'>
<class '__main__.NewStyleClass'> <class '__main__.NewStyleClass'>
經典類的繼承是深度優先,新式類的繼承是廣度優先。
####################################################################
子類和派生的動機是為了節省開發的事件,更加便捷的進行相關的設計。
OOP的更強大功能之一是能夠使用一個已經定義好的類,擴展它或者對其進行修改,而不會影響系統中使用現有類得其他代碼片段。
OOD允許類特征在子孫類或子類中進行集成。這些子類從基類(或者稱作祖先類、超類)繼承他們得核心屬性。而且,這些派生可能會擴展到多代。
在一個層次的派生關系中的相關類(或者是在類樹圖種垂直相鄰)是父類和子類關系。從同一個父類派生出來得這些類(或者是在類樹中水平響鈴)是同胞關系。父類和所有高層類都被認為是祖先。
繼承描述了,基類的屬性如何“遺傳”給派生類。一個子類可以繼承它的基類的任何屬性,不管是數據屬性還是方法。
對於類、函數\方法,還有模塊來說,文檔字符串都是唯一的,所以特殊屬性__doc__不會從基類中繼承過來。、
繼承覆蓋方法:super()內建方法;
使用super()的漂亮指出在於,你不需要明確給出任何基類名字。。。,它都可以幫你干了!使用super()的重點,使你不需要明確提供父類。這意味着如果你改變了類繼承關系,你只需要改一行代碼(class語句本身)而不必在大量代碼中去查找所有被修改的那一個類的名字。
多重繼承:
python和C++一樣,支持多繼承。概念雖然容易,但是困難的工作是如果子類調用一個自身沒有定義的屬性,它是按照何種順序去到父類尋找呢,尤其是眾多父類中有多個都包含該同名屬性。
對經典類和新式類來說,屬性的查找順序是不同的。現在我們分別看一下經典類和新式類兩種不同的表現:
經典類:
#! /usr/bin/python # -*- coding:utf-8 -*- class P1(): def foo(self): print 'p1-foo' class P2(): def foo(self): print 'p2-foo' def bar(self): print 'p2-bar' class C1(P1,P2): pass class C2(P1,P2): def bar(self): print 'C2-bar' class D(C1,C2): pass if __name__ =='__main__': d=D() d.foo() d.bar()
執行的結果:
p1-foo
p2-bar
將代碼實例,畫了一個圖,方便理解:
從上面經典類的輸出結果來看,
實例d調用foo()時,搜索順序是 D => C1 => P1,
實例d調用bar()時,搜索順序是 D => C1 => P1 => P2
總結:經典類的搜索方式是按照“從左至右,深度優先”的方式去查找屬性。d先查找自身是否有foo方法,沒有則查找最近的父類C1里是否有該方法,如果沒有則繼續向上查找,直到在P1中找到該方法,查找結束。
新式類:
#! /usr/bin/python # -*- coding:utf-8 -*- class P1(object): def foo(self): print 'p1-foo' class P2(object): def foo(self): print 'p2-foo' def bar(self): print 'p2-bar' class C1(P1,P2): pass class C2(P1,P2): def bar(self): print 'C2-bar' class D(C1,C2): pass if __name__ =='__main__':
print D.__mro__ #只有新式類有__mro__屬性,告訴查找順序是怎樣的 d=D() d.foo() d.bar()
執行的結果:
(<class '__main__.D'>, <class '__main__.C1'>, <class '__main__.C2'>, <class '__main__.P1'>, <class '__main__.P2'>, <type 'object'>)
p1-foo
C2-bar
從上面新式類的輸出結果來看,
實例d調用foo()時,搜索順序是 D => C1 => C2 => P1
實例d調用bar()時,搜索順序是 D => C1 => C2
總結:新式類的搜索方式是采用“廣度優先”的方式去查找屬性。