一.繼承介紹
1? ? 什么是繼承
? ??????繼承一種新建類的方式,新建的類稱之為子類/派生類,被繼承的類稱之為父類\基類\超類
? ??????python中繼承的特點:
? ??????????????1. 子類可以遺傳/重用父類的屬性
????????????????2. python中一個子類可以同時繼承多個父類
????????????????3. 在繼承背景下去說,python中的類分為兩種:新式類,經(jīng)典類
????????新式類: 但凡繼承了object的類Foo,以及該類的子類...都是新式類
????????????????在python3中一個類即便是沒有顯式地繼承任何類,默認(rèn)就會繼承object
????????????????即python3中所有的類都是新式類
????????經(jīng)典類:沒有繼承object的類,以及該類的子類...都是經(jīng)典類
????????????????在python2中才區(qū)分新式類與經(jīng)典類,
? ? ????????? ? 在python2中一個類如果沒有顯式地繼承任何類,也不會繼承object
二.用繼承的目的:減少代碼冗余
例如上節(jié)的OldboyStudent類,如果里面加上成績(score)屬性
再定義一個老師類(OldboyTeacher),里面加上老師的等級(level)
這樣兩個類里面都會出現(xiàn)self.name=name\self.age=age....
因此,我們可以像提取對象的相同屬性來定義類一樣來提取類的相同屬性定義一個父類
例如:
? ? ? ? 貓可以:喵喵叫? 吃? 喝? 拉? 撒
? ???????貓可以:汪汪叫? 吃? 喝? 拉? 撒
如果我們要分別為貓和狗創(chuàng)建一個類,那么就需要為 貓 和 狗 實現(xiàn)他們所有的功能左胞,偽代碼如下:
class 貓:
????def 喵喵叫(self):
????????print '喵喵叫'
????def 吃(self):
????????# do something
????def 喝(self):
????????# do something
????def 拉(self):
????????# do something
????def 撒(self):
????????# do something
class 狗:
????def 汪汪叫(self):
????????print '汪汪叫'
????def 吃(self):
????????# do something
????def 喝(self):
????????# do something
????def 拉(self):
????????# do something
????def 撒(self):
????????# do something
上述代碼不難看出蔗候,吃暴心、喝眶蕉、拉闻牡、撒是貓和狗都具有的功能冠胯,而我們卻分別的貓和狗的類中編寫了兩次阀蒂。如果使用 繼承 的思想,如下實現(xiàn):
動物:吃喝拉撒
? ? 貓:喵喵叫(貓繼承動物的功能)?
????狗:汪汪叫(狗繼承動物的功能)
偽代碼如下:
class 動物:
? ? def 吃(self):
? ? ? ? # do something
????def 喝(self):
? ? ? ? # do something
????def 拉(self):
? ? ? ? # do something
????def 撒(self):
? ? ? ? # do something
# 在類后面括號中寫入另外一個類名酌心,表示當(dāng)前類繼承另外一個類
class 貓(動物):
? ? def 喵喵叫(self):
? ? ? ? print'喵喵叫'
class 狗(動物):
? ? def 汪汪叫(self):
? ? ? ? print'喵喵叫'
代碼實現(xiàn):
class Animal:
? ? def eat(self):
? ? ? ? print("%s 吃 "%self.name)
? ? def drink(self):
? ? ? ? print("%s 喝 "%self.name)
? ? def shit(self):
? ? ? ? print("%s 拉 "%self.name)
? ? def pee(self):
? ? ? ? print("%s 撒 "%self.name)
class Cat(Animal):
? ? def__init__(self, name):
? ? ? ? self.name = name
? ? ? ? self.breed ='貓'
????def cry(self):
? ? ? ? print('喵喵叫')
class Dog(Animal):
? ? def__init__(self, name):
? ? ? ? self.name = name
? ? ? ? self.breed='狗'
????def cry(self):
? ? ? ? print('汪汪叫')
# ######### 執(zhí)行 #########
c1 = Cat('小白家的小黑貓')
c1.eat()
c2 = Cat('小黑的小白貓')
c2.drink()
d1 = Dog('胖子家的小瘦狗')
d1.eat()
屬性的查找
class Foo:
????def f1(self):
????????print('Foo.f1')
????def f2(self):
????????print('Foo.f2')
????????self.f1()
class Bar(Foo):
????def f1(self):
????????print('Bar.f1')
b=Bar()
b.f2()
'''
輸出結(jié)果:
Foo.f2
Bar.f1
'''
三.在子類派生的新方法中重用父類功能的方式一
在子類派生出的新方法中重用父類功能的方式一:
指名道姓地引用某一個類中的函數(shù)
總結(jié):
1. 與繼承無關(guān)
2. 訪問是類的函數(shù),沒有自動傳值的效果
class OldboyPeople:
????school='oldboy'
????def __init__(self,name,age,sex):
????????self.name=name
????????self.age=age
????????self.sex=sex
class OldboyStudent(OldboyPeople):
????def __init__(self,name,age,sex,score):
????????OldboyPeople.__init__(self,name,age,sex)
????????self.score=score
????????def choose_course(self):
????????????print('%s 正在選課'% self.name)
class OldboyTeacher(OldboyPeople):
????def __init__(self,name,age,sex,leval):
????OldboyPeople.__init__(self,name,age,sex)
????self.leval=leval
????def paly_score(self,stu,score):
? ??????stu.score=score
????????print('%s 老師給%s同學(xué)打了%s分'%(self.name,stu.name,score))
stu1=OldboyStudent('wzj','18','male',100)
print(stu1.__dict__)
stu1.choose_course()
tea=OldboyTeacher('egon','18','male',10)
print(tea.__dict__)
tea.paly_score(stu1,720)
print(stu1.score)
四 屬性查找
1.在單繼承背景下屬性的查找優(yōu)先級:對象->對象的類->父類->父類.....
2.在多繼承背景下屬性的查找優(yōu)先級:
? ? ? ? 1).如果一個子類繼承多個分支(多個分支沒有共同繼承一個非object的類)
? ? ? ? ????此時屬性的查找優(yōu)先級是:對象->對象的類->按照從左往右的順序一個分支一個分支的找下去
? ? ? ? ? ? 如下圖,先從A的實例化的對象中找,如果沒有,按照下圖所示數(shù)字順序查找
菱形繼承問題:
新式類 : 廣度優(yōu)先查找,從左往右一個分支一個分支的查找,在最后一個分支才去查找頂級類
????????????????obj->A->B->E->C-F->D->G->object
?經(jīng)典類 : 深度優(yōu)先查找,從左往右一個分支一個分支的查找,在第一個分支就查找頂級類
? ??????????????obj->A->B->E->G->C-F->D
mro:用來查看查找順序的方法
print(A.mro())
#輸出結(jié)果::
#[<class '__main__.A'>,<class'__main__.B'>, <class '__main__.E'>, <class '__main__.C'>, <class'__main__.F'>, <class '__main__.D'>, <class '__main__.G'>, <class'object'>]
六 .在子類派生出的新方法中重用父類功能的方式二
super()? ? ? 必須在類中用
在python2中:super(自己的類名,自己的對象)
在python3中:super()
調(diào)用該函數(shù)會得到一個特殊的對象,該對象專門用來訪問父類中的屬性,!!!完全參照mro列表!!!!
總結(jié):
1. 嚴(yán)格依賴?yán)^承的mro列表
2. 訪問是綁定方法,有自動傳值的效果
class A:
????def f1(self):
????????print('A.f1')
????????super().f2()
class B:
????def f2(self):
????????print('B.f2')
class C(A,B):
????def f2(self):
????????print('C.f2')
obj=C()
# print(C.mro())
obj.f1()#===>A.f1? ? ?B.f2
這里調(diào)用obj.f1(),找到了A類下的f1,打印了'A.f1 ',再執(zhí)行了super().f2(),那么這個f2()是找B中的還是C中的
先看對象所在類的mro表為:[<class '__main__.C'>, <class'__main__.A'>, <class '__main__.B'>, <class 'object'>]
A中的super,所以找A的父類,這里的父類并不是明顯的父類,二十mro表中A后面的類,即B,所以這里打印的是B.f2