现在创造一个cat实例,由于子类没有自己的构造函数,就会调用到父类的构造函数。
class Mammal:
def __init__(self,name,sex):
self.name = name
self.sex = sex
self.num_eyes = 2
def breath(self):
print(self.name + "在呼吸...")
def poop(self):
print(self.name + "在拉屎")
class Human(Mammal):
def read(self):
print(self.name + "在阅读")
class Cat(Mammal):
def scratch_sofa(self):
print(self.name + "在抓沙发...")
def poop(self):
print(self.name + "在猫砂上拉屎...")
cat1 = Cat("Jojo","男")
cat1.poop()
如果子类有自己的拉屎方法的话,就会调用自己的。这背后的逻辑是:
优先看所属的类有没有该方法,没有的话往上找父类的同名方法用。
但如果像这样,给子类写init方法,那创建子类实例时,就会优先调用子类的构造函数。导致实例只有has_tail属性。
class Cat(Mammal):
def __init__(self):
self.has_tail = True
def scratch_sofa(self):
print(self.name + "在抓沙发...")
一个方法是在子类__init__里面,把name、sex等属性也全部都写上,但这就会造成重复代码,更优雅的做法是,用super()这个方法,super()会返回当前类的父类,所以在子类__init__方法里面,写super().__init__(),这就会调用父类的构造函数。那么子类也会有姓名、性别、眼睛数量的属性了。所以通过继承我们成功把重复的属性和方法代码行给移除了,并且在实例化、获得属性、调用方法时和之前都一模一样,没有任何区别。
class Cat(Mammal):
def __init__(self,name,sex):
super().__init__(name,sex)
self.has_tail = True
def scratch_sofa(self):
print(self.name + "在抓沙发...")
cat1 = Cat("Jojo","男")
print(cat1.name)
在子类下面调用super方法,super()会返回当前类的父类,所以在子类的__init__方法里,写super().__init__()就会调用父类的构造函数。子类也会有姓名、性别、眼睛和数量的属性了。通过继承,我们成功把重复的属性和方法代码行给移除了。