多继承
多继承
概念
- 子类 可以拥有 多个父类,并且具有 所有父类 的 属性 和 方法
- 例如:孩子 会继承自己 父亲 和 母亲 的 特性
语法
class 子类名(父类名1, 父类名2...)
pass
class A():
def test(self):
print("这是A类的方法")
class B():
def demo(self):
print("这是B类的方法")
class C(B, A):
pass
c = C()
# 在测试时可能会调用放在前面的父类的方法,但不要以为这是按顺序调用的
c.test()
c.demo()
运行结果:
1 多继承的使用注意事项
问题的提出
- 如果 不同的父类 中存在 同名的方法,子类对象 在调用方法时,会调用 哪一个父类中的方法呢?
提示:开发时,应该尽量避免这种容易产生混淆的情况! —— 如果 父类之间 存在 同名的属性或者方法,应该 尽量避免 使用多继承
Python 中的 MRO —— 方法搜索顺序
-
Python
中针对 类 提供了一个 内置属性__mro__
可以查看 方法 搜索顺序 - MRO 是
method resolution order
,主要用于 在多继承时判断 方法、属性 的调用 路径
print(C.__mro__)
class A():
def test(self):
print("这是A类的方法")
class B():
def test(self):
print("这是B类的方法")
class C(B, A):
pass
c = C()
# 在测试时可能会调用放在前面的父类的方法,但不要以为这是按顺序调用的
c.test()
c.test()
# c类对象调用方法的顺序
print(C.__mro__)
# 注意此处的类为类名
对于print(C.__mro__)的输出结果为:
为什么先是B类在是A类呢?
来看下我们在继承时类的顺序
我们先写的B类后写的A类
我们来对这个结果进行剖析下
输出结果为元祖,当C类创建的对象调用方法时会首先在本类中查找有没有要调用的方法,如果有就直接调用执行而不会向后搜索, 如果没有则按照元祖的顺序继续向后从第二个类中查找有没有要调用的方法,若有则调用并不在向后搜索,若没有则继续按顺序向下寻找, 最后的一个类为object类,在Python3中object是所有类的父类
- 在搜索方法时,是按照
__mro__
的输出结果 从左至右 的顺序查找的 - 如果在当前类中 找到方法,就直接执行,不再搜索
- 如果 没有找到,就查找下一个类 中是否有对应的方法,如果找到,就直接执行,不再搜索
- 如果找到最后一个类,还没有找到方法,程序报错
2 新式类与旧式(经典)类
object
是Python
为所有对象提供的 基类,提供有一些内置的属性和方法,可以使用dir
函数查看
在ipython中我们没有在A类中定义任何方法,但使用dir()函数查看却有很多内置放方法,这些内置方法都是A类的父类object提供的
-
新式类:以
object
为基类的类,推荐使用 -
经典类:不以
object
为基类的类,不推荐使用 -
在
Python 3.x
中定义类时,如果没有指定父类,会 默认使用object
作为该类的 基类 ——Python 3.x
中定义的类都是 新式类 -
在
Python 2.x
中定义类时,如果没有指定父类,则不会以object
作为 基类
新式类 和 经典类 在多继承时 —— 会影响到方法的搜索顺序
为了保证编写的代码能够同时在 Python 2.x
和 Python 3.x
运行!
今后在定义类时,如果没有父类,建议统一继承自 object
class 类名(object):
pass