python子類會繼承父類所有的類屬性和類方法。嚴格來說,類的構造方法其實就是實例方法,因此,父類的構造方法,子類同樣會繼承。
我們知道,python是一門支持多繼承的面向對象編程語言,如果子類繼承的多個父類中包含同名的類實例方法,則子類對象在調用該方法時,會優先選擇排在最前面的父類中的實例方法。顯然,構造方法也是如此。
class People:
def __init__(self, name):
self.name = name
def say(self):
print("我是人,名字為:", self.name)
class Animal:
def __init__(self, food):
self.food = food
def display(self):
print("我是動物,我吃", self.food)
#People中的name屬性和say()會覆蓋Animal類中的
class Person(People, Animal):
pass
xiaobai = Person("小白")
#Person類同時繼承People和Animal,其中People在前。
#這意味着,在創建對象xiaobai時,其將會調用從People類繼承來的構造函數,因此上面程序創建xiaobai對象的同時,還要給name屬性進行賦值。
xiaobai.say()
xiaobai.display()
我是人,名字為: 小白
Traceback (most recent call last):
AttributeError: 'Person' object has no attribute 'food'
#從Animal類中繼承的display()方法中,需要用到food屬性的值,
#但由於People類的構造方法覆蓋了Animal類的構造方法,使得在創建xiaobai對象時,Animal類的構造方法未得到執行,所以程序出錯。
針對這種情況,正確的做法是定義Person類自己的構造方法(等同於重寫第一個直接父類的構造方法),但是需要注意的是,在子類中定義構造方法,則必須在該方法中調用父類的構造方法。
- 未綁定方式
- 使用super()函數,但如果涉及多繼承,該函數只能調用第一個直接父類的構造方法。
也就是說,涉及到多繼承時,在子類構造函數中,調用第一個父類構造方法的方式有以上兩種,而調用其他父類構造方法的方式只能使用未綁定方法。
class People:
def __init__(self, name):
self.name = name
def say(self):
print("我是人,名字為:", self.name)
class Animal:
def __init__(self, food):
self.food = food
def display(self):
print("我是動物,我吃", self.food)
class Person(People, Animal):
#自定義構造方法
def __init__(self, name, food):
#調用第一個父類的構造方法
super().__init__(name)
#People.__init__(self, name)使用未綁定方法調用第一個父類的構造方法
#調用其他父類的構造方法,需要手動給self傳值
Animal.__init__(self, food)
xiaobai = Person("xiaobai", "熟食")
xiaobai.say()
xiaobai.display()
我是人,名字為: xiaobai
我是動物,我吃 熟食
可以看到,Person類自定義的構造方法中,調用People類構造方法,可以使用super()函數,也可以使用未綁定方法。但是調用Animal類的構造方法,只能使用未綁定方法。
