Python 多繼承中的一個(gè)詭異現(xiàn)象 既是 Father又是grandfather
我們知道,在面向?qū)ο缶幊汤锩妫?繼承 是一個(gè)很重要的概念。子類(lèi)可以使用父類(lèi)的方法和屬性。
例如下面這段代碼:
class Father:
def __init__(self):
self.address = '上海'
def say(self):
print('我是爸爸')
class Son(Father):
def __init__(self):
super().__init__()
def say(self):
print('我是兒子')
son = Son()
print(son.address)
運(yùn)行效果如下圖所示:

從圖中可以看到,子類(lèi)并沒(méi)有 self.address 這個(gè)屬性,但是當(dāng)我們直接打印的時(shí)候,并不會(huì)報(bào)錯(cuò),它會(huì)自動(dòng)使用父類(lèi)的 address 屬性。
顯然,如果一個(gè)屬性,子類(lèi)也沒(méi)有,父類(lèi)也沒(méi)有,那肯定會(huì)報(bào)錯(cuò),如下圖所示:

我們也知道,Python 是支持多繼承的,一個(gè)子類(lèi)可以有多個(gè)父類(lèi)。那么,大家請(qǐng)看下面這段代碼:
class GrandFather:
def __init__(self):
self.address = '上海'
def say(self):
print('我是爸爸')
class Father:
def __init__(self):
self.age = 100
def where(self):
print('我現(xiàn)在住在:', self.address)
class Son(GrandFather, Father):
def __init__(self):
super().__init__()
def say(self):
print('我是兒子')
son = Son()
son.where()
運(yùn)行效果如下圖所示:

大家仔細(xì)觀察,會(huì)發(fā)現(xiàn)這段代碼有點(diǎn)奇怪。我調(diào)用的是 son.where() 方法,由于 Son 類(lèi)沒(méi)有這個(gè)方法,于是它會(huì)去它的兩個(gè)父類(lèi)里面找。于是在 Father 這個(gè)父類(lèi)里面找到了。于是執(zhí)行 Father 里面的 where() 方法,目前為止沒(méi)有問(wèn)題。
但接下來(lái)就不對(duì)了, .where() 方法里面,調(diào)用了 self.address 屬性??蓡?wèn)題是 Father 這個(gè)類(lèi)它并沒(méi)有 .address 屬性??!而且 Father 也沒(méi)有父類(lèi),那么這個(gè) .address 屬性是從哪里來(lái)的?
難道說(shuō),在開(kāi)發(fā)者不知道的隱秘的角落里面, GrandFather 類(lèi)悄悄成為了 Father 的父類(lèi)?這樣一來(lái), GrandFather 豈不是又是 C 的父類(lèi),又是 C 的父類(lèi)的父類(lèi)? GrandFather 既是爸爸又是爺爺?
實(shí)際上,并不存在這么混亂的關(guān)系。要解釋這個(gè)現(xiàn)象,我們就要從 self 這個(gè)東西說(shuō)起。
我們知道,類(lèi)的屬性都是以 self 開(kāi)頭,方法的第一個(gè)參數(shù)也是 self 。那么這個(gè) self 到底是什么東西?我們用一段小代碼來(lái)看看它是什么東西:
class A:
def get_self(self):
return self
test = A()
what_is_self = test.get_self()
test is what_is_self
運(yùn)行效果如下圖所示:

從圖里面可以看到, self 實(shí)際上就是這個(gè)類(lèi)的實(shí)例。我們?cè)賮?lái)看有繼承的情況:
class A:
def get_self(self):
return self
class B(A):
def __init__(self):
...
test = B()
what_is_self = test.get_self()
print(what_is_self)

從圖中可以看到,雖然我在 A 類(lèi)的 .get_self() 方法中返回了 self ,但這個(gè) self 實(shí)際上是 B 類(lèi)的實(shí)例。因?yàn)槲易允贾两K就只初始化了 B 類(lèi),并沒(méi)有初始化 A 類(lèi)。A 雖然是 B 類(lèi)的父類(lèi)。但父類(lèi)的 self 都會(huì)變成子類(lèi)的實(shí)例。
明白這一點(diǎn)以后,前面的問(wèn)題就很好解釋了,我們多打印一些信息:

大家注意畫(huà)紅線(xiàn)的地方, self 始終都是 Son 類(lèi)的實(shí)例。所以,一開(kāi)始初始化 .address 的時(shí)候,就是初始化的 Son 的實(shí)例的 .address 屬性。后面在 .where 里面調(diào)用 .address 的時(shí)候,也是讀取的 Son 的實(shí)例的 .address 屬性。所以,并不存在 Father 類(lèi)去讀 GrandFather 類(lèi)的情況。自始至終,都是 Son 類(lèi)的實(shí)例在進(jìn)行各種操作。
所以,在這個(gè)例子里面,當(dāng)使用了繼承以后,所有父類(lèi)的屬性和方法,子類(lèi)如果有相同的名字,那么以子類(lèi)的為準(zhǔn)。如果子類(lèi)沒(méi)有定義,那么父類(lèi)的屬性和方法,其實(shí)都會(huì)跑到子類(lèi)里面去。 所有看起來(lái)是父類(lèi)進(jìn)行的操作,其實(shí)都是子類(lèi)在進(jìn)行 。上面的代碼,甚至可以近似等價(jià)于:

由于 say 方法在子類(lèi)中有了定義,所以子類(lèi)覆蓋父類(lèi)。以子類(lèi)的 say 方法為準(zhǔn)。 where 和 address 由于子類(lèi)沒(méi)有定義,所以 Father 類(lèi)的 where 方法和 GrandFather 里面的 address 屬性,都會(huì)直接 跑 到子類(lèi)里面。
到此這篇關(guān)于Python 多繼承中的一個(gè)詭異現(xiàn)象 既是 Father又是grandfather的文章就介紹到這了,更多相關(guān)Python 多繼承中的一個(gè)詭異現(xiàn)象 內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
PyTorch變分自編碼器的構(gòu)建與應(yīng)用小結(jié)
變分自編碼器是一種強(qiáng)大的深度學(xué)習(xí)模型,用于學(xué)習(xí)數(shù)據(jù)的潛在表示并能生成新的數(shù)據(jù)點(diǎn),使用PyTorch實(shí)現(xiàn)VAE不僅可以加深對(duì)生成模型的理解,還可以利用其靈活性進(jìn)行各種實(shí)驗(yàn),這篇文章主要介紹了PyTorch變分自編碼器的構(gòu)建與應(yīng)用,需要的朋友可以參考下2024-07-07
python里的單引號(hào)和雙引號(hào)的有什么作用
在本篇文章里小編給大家分享的是一篇關(guān)于python里的單引號(hào)和雙引號(hào)的作用的相關(guān)內(nèi)容,需要的朋友們可以學(xué)習(xí)下。2020-06-06
Linux添加Python?path方法及修改環(huán)境變量的三種方法
這篇文章主要介紹了Linux添加Python?path方法及修改環(huán)境變量的三種方法,Linux 下設(shè)置環(huán)境變量有三種方法,一種用于當(dāng)前終端,一種用于當(dāng)前用戶(hù),一種用于所有用戶(hù),本文對(duì)每種方法給大家介紹的非常詳細(xì),需要的朋友可以參考下2022-07-07
Python列表中多元素刪除(移除)的實(shí)現(xiàn)
本文主要介紹了Python列表中多元素刪除(移除)的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-03-03
pandas讀取中文xlsx文件出現(xiàn)的問(wèn)題
這篇文章主要介紹了pandas讀取中文xlsx文件出現(xiàn)的問(wèn)題及解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-05-05

