单继承
- 单继承就是某个类只继承自一个父类,同时,继承关系中可以有多级继承
- 继承过程中,子类可以使用父类的所有非私有属性或方法
- 如果父类或更高级的父类,实现了init方法,并且进行了参数设定,实例化子类对象时必须传值
# 单继承:一个子类,只继承一个父类,并且可以多级继承
# 定义一个Person类
class Person(object):
def __init__(self, name, age):
self.name = name
self.__age = age
# 定义一个Father类,继承Person
class Father(Person):
def __sing(self):
print('我会唱学猫叫,跟我一起来')
def dance(self):
print('我会跳四小天鹅,就是天鹅还缺仨')
# 定义一个Son类,继承Father
class Son(Father):
def play(self):
# AttributeError: 'Son' object has no attribute '_Son__sing'
# 继承父类时,只能继承父类中的非私有属性和方法
self.__sing()
self.dance()
# 实例化一个Son对象
# TypeError: __init__() missing 2 required positional arguments: 'name' and 'age'
# 应为son继承了father father继承了person, 在person中书写了init方法的参数,所以此处必须传参
# s1 = Son()
s1 = Son('xiaoming', 12)
# s1 继承了父类的属性和方法,在Son类中我们没有书写任何内容,但是可以调用父类及其父类的父类中的方法
# s1.sing()
# 调用方法时如果父类中书写了 我们就可以调用到,但是父类中的私有属性或者方法,我们无法调用
# AttributeError: 'Son' object has no attribute '__age'
# print(s1.__age)
# AttributeError: 'Son' object has no attribute '__sing'
# s1.__sing()
# s1.play()
# 结论:
# 1.在继承中可以多级继承,子类中可以使用父类及父类的父类中非私有的属性和方法
# 2.如果在父类或者更高级的父类中实现了init方法,并且书写了参数,则实例化对象时,必须传值
# 扩展:
# 怎样查询类的继承链条
# (<class '__main__.Son'>, <class '__main__.Father'>, <class '__main__.Person'>, <class 'object'>)
# 使用类名.__mro__可以输出类的继承链条,同时这个顺序也是方法或属性查找的顺序
print(Son.__mro__)
多继承
- 一个子类,继承多个父类的过程就是多继承
- 在多继承中,子类可以调用多个父类中的非私有方法或者属性
- 多继承中,如果出现同名属性或方法,优先调用继承位置靠前的父类中的方法或属性
# 多继承:一个类定义时,继承了多个父类,同时可以使用多个父类中的方法或者属性
# 格式: class 子类名(父类名1, 父类名2):
class Father(object):
def dance(self):
print('我现在要跳一个舞,赶紧出去')
def sing(self):
print('我要唱一个学猫叫,一起来')
class Mother(object):
def dance(self):
print('我现在要跳一个广场舞,离我远点,不然摔倒了赖你')
def sing(self):
print('我要唱一个大河向东流,赶紧跑')
# 同时继承两个父类,则在使用子类对象时可以调用两个父类中的所有非私有属性和方法
# class Son(Father, Mother):
# pass
#
#
# s1 = Son()
#
# s1.dance()
# s1.sing()
# 疑问: 如果两个父类中有重名方法该怎么办?
# 在下述情况下,只会调用Father类中的sing和dance方法
# class Son(Father, Mother):
# pass
#
#
# s1 = Son()
#
# s1.dance()
# s1.sing()
# 如果我将两个父类的顺序进行调换
# 此时,只会调用Mother类中的sing 和dance方法
class Son(Mother, Father):
pass
s1 = Son()
s1.dance()
s1.sing()
# 结论:如果存在同名方法,在继承时,谁的继承位置更靠前就调用谁内部的代码
子类中重写父类方法
- 子类中重写父类方法,则调用方法时,直接调用子类中的方法,不会调用父类的
- 重写时只要方法名称相等即可,不需要进行参数的校对
- 为什么可以重写父类方法,因为在调用方法或者属性时,会按照继承层级依次查找
# 定义一个Person类
class Person(object):
def __init__(self, name, age):
self.name = name
self.age = age
# 定义一个Father类,继承Person
class Father(Person):
def sing(self):
print('我会唱学猫叫,跟我一起来')
def dance(self):
print('我会跳四小天鹅,就是天鹅还缺仨')
# 定义一个Son类,继承Father
class Son(Father):
# 需求:在Son执行sing方法时,我么你让他唱一分钱
def sing(self):
print('我喜欢唱一分钱, 你自己学猫叫吧')
# 我们进行方法 重写的时候,不需要关注参数,只需方法名相同即可
def __init__(self):
pass
# s1 = Son('xiaoming', 12)
# s1.sing()
# 为什么子类中重写了父类方法就不能进行调用了?
# 之前我么讲了__mro__ 打印继承顺序,同时其也是方法或属性的调用顺序,例如想使用Son对象调用sing方法,但是不知道sing在哪个类中
# 所以,系统先去当前Son类中查找,查看是否存在sing方法
# 如果存在,则调用,如果不存在,则去父类中 Father中查找,如果Father类中存在sing则调用,如果不存在,则去更高级父类(person)中查找
# 直到查询到object类中,如果依然不存在,则报错
# 所以如果子类中书写了对应的方法,则父类中的同名方法无法被调用
# 可不可以让Son类不需要使用name和age就可以创建对象呢? 在Son中重写init方法
s1 = Son()
s1.sing()