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类的构造方法,只能使用未绑定方法。