本文主要做科普用,在真实编程中不建议使用多重继承,或者少用多重继承,避免使代码难以理解。

方法解析顺序(MRO)

关于多重继承,比较重要的是它的方法解析顺序(可以理解为类的搜索顺序),即MRO。这个跟类是新式类还是经典类有关,因为两者的搜索算法不同。

在Python2及以前的版本,由任意内置类型派生出的类(只要一个内置类型位于类树的某个位置),都属于新式类;反之,不由任意内置类型派生出的类,则称之为经典类

在Python3以后,没有该区分,所有的类都派生自内置类型object,不管有没有显式继承object,都属于新式类

对于经典类,多重继承的MRO是深度优先,即从下往上搜索;新式类的MRO是采用C3算法(不同情况下,可表现为广度优先,也可表现为深度优先)

C3算法表现为深度优先的例子:

# C3-深度优先(D -> B -> A -> C)
class A:
var = 'A var' class B(A):
pass class C:
var = 'C var' class D(B, C):
pass if __name__ == '__main__':
# [<class '__main__.D'>, <class '__main__.B'>, <class '__main__.A'>, <class '__main__.C'>, <class 'object'>]
print(D.mro())
# A var
print(D.var)
复制代码

C3算法表现为广度优先的例子:

# C3-广度优先(D -> B -> C -> A)
class A:
var = 'A var' class B(A):
pass class C(A):
var = 'C var' class D(B, C):
pass if __name__ == '__main__':
# [<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.A'>, <class 'object'>]
print(D.mro())
# C var
print(D.var) 复制代码

注:关于C3的详细算法本文不讨论,因为我也搞不懂(狗头保命)

菱形多重继承

其实菱形多重继承上面已经有例子了,就是C3算法表现为广度优先这个例子,画起图来是这样的:

菱形多重继承会导致的一个问题是A初始化了两次,如下:

class A:
def say(self):
print("A say") class B(A):
def say(self):
print("B say")
A.say(self) class C(A):
def say(self):
print("C say")
A.say(self) class D(B, C):
def say(self):
print("D say")
B.say(self)
C.say(self) if __name__ == '__main__':
dd = D()
dd.say() 复制代码

如果只想调用一次A,可使用super方法:

class A:
def say(self):
print("A say") class B(A):
def say(self):
print("B say")
super().say() class C(A):
def say(self):
print("C say")
super().say() class D(B, C):
def say(self):
print("D say")
super().say() if __name__ == '__main__':
print(D.mro())
dd = D()
dd.say() 复制代码

_init_ 与 super()

1.如果父类有init方法,子类没有,则子类默认继承父类的init方法

class A:
def __init__(self, a1, a2):
self.a1 = a1
self.a2 = a2 def say(self):
print("A say, a1: %s, a2: %s" % (self.a1, self.a2)) class B(A):
def say(self):
print("B say, a1: %s, a2: %s" % (self.a1, self.a2)) if __name__ == '__main__':
# 因为B继承了A的init方法,所以也要传入 a1,a2参数
bb = B("10", "100")
bb.say() 复制代码

2.如果父类有init方法,子类也有,可理解为子类重写了父类的init方法

class A:
def __init__(self, a1, a2):
self.a1 = a1
self.a2 = a2 def say(self):
print("A say, a1: %s, a2: %s" % (self.a1, self.a2)) class B(A):
def __init__(self, b1):
self.b1 = b1 def say(self):
print("B say, b1: %s" % self.b1) if __name__ == '__main__':
# 此处B重写了A的init方法,所以只需要传入b1参数,也没有拥有a1,a2属性
bb = B("10")
bb.say() 复制代码

3.对于第二点,为了能使用或者扩展父类的行为,更常见的做法是在重写init方法的同时,显示调用父类的init方法(意味传的参数要变成3个)。

# 三种写法
class A:
def __init__(self, a1, a2):
self.a1 = a1
self.a2 = a2 def say(self):
print("A say, a1: %s, a2: %s" % (self.a1, self.a2)) class B(A):
def __init__(self, b1, a1, a2):
# 第一种写法: Python2的写法
# super(B, self).__init__(a1, a2)
# 第二种写法(推荐):Python3的写法,与第一种等价
super().__init__(a1, a2)
# 第三种写法:与前两种等价,不过这种需要显示调用基类,而第二种不用
# A.__init__(self, a1, a2)
self.b1 = b1 def say(self):
print("B say, a1: %s, a2: %s, b1: %s" % (self.a1, self.a2, self.b1)) if __name__ == '__main__':
# 此处B重写了A的init方法,所以只需要传入b1参数,也没有拥有a1,a2属性
bb = B("10", "100", "1000")
bb.say()
复制代码

最后的提醒

注意 __init__ 方法不要写错,避免写成__ini__或者其他的,因为两个是在太像,出了问题很难排查(坑过两次)。

本文的文字及图片来源于网络,仅供学习、交流使用,不具有任何商业用途,如有问题请及时联系我们以作处理

想要获取更多Python学习资料可以加QQ:2955637827私聊或加Q群630390733大家一起来学习讨论吧!

小白都能理解的Python多继承的更多相关文章

  1. 图解Go协程调度原理,小白都能理解

    阅读本文仅需五分钟,golang协程调度原理,小白也能看懂,超实用. 什么是协程 对于进程.线程,都是有内核进行调度,有CPU时间片的概念,进行抢占式调度.协程,又称微线程,纤程.英文名Corouti ...

  2. 小白都能理解的TCP三次握手四次挥手

    前言 TCP在学习网络知识的时候是经常的被问到知识点,也是程序员必学的知识点,今天小杨用最直白的表述带大家来认识认识,喜欢的朋友记得点点关注哈. 何为TCP 上点官方的话:是一种面向连接(连接导向)的 ...

  3. python基础——继承和多态

    python基础——继承和多态 在OOP程序设计中,当我们定义一个class的时候,可以从某个现有的class继承,新的class称为子类(Subclass),而被继承的class称为基类.父类或超类 ...

  4. python类继承

    面向对象的编程带来的主要好处之一是代码的重用,实现这种重用的方法之一是通过 继承 机制.继承完全可以理解成类之间的 类型和子类型 关系. 假设你想要写一个程序来记录学校之中的教师和学生情况.他们有一些 ...

  5. sqlalchemy mark-deleted 和 python 多继承下的方法解析顺序 MRO

    sqlalchemy mark-deleted 和 python 多继承下的方法解析顺序 MRO 今天在弄一个 sqlalchemy 的数据库基类的时候,遇到了跟多继承相关的一个小问题,因此顺便看了一 ...

  6. Python进阶-继承中的MRO与super

    Python进阶-继承中的MRO与super 写在前面 如非特别说明,下文均基于Python3 摘要 本文讲述Python继承关系中如何通过super()调用"父类"方法,supe ...

  7. python基础——继承实现的原理

    python基础--继承实现的原理 1 继承顺序 class A(object): def test(self): print('from A') class B(A): def test(self) ...

  8. python基础——继承与派生、组合

    python基础--继承与派生 1 什么是继承: 继承是一种创建新的类的方式,在python中,新建的类可以继承自一个或者多个父类,原始类成为基类或超累,新建的类成为派生类或子类 1.1 继承分为:单 ...

  9. python 面向对象 继承 派生 组合

    具体参考博客:http://www.cnblogs.com/linhaifeng/articles/6182264.html#_label12 一.面向对象 面向对象:对象与对象之间的相互交互,不可预 ...

随机推荐

  1. H5系列之新input

    虽说H5 新填了几个很方便的input 类型,但是吧,想法是美好的,但是现实很残酷,兼容性不太好.基本只有google浏览器能用.但是既然出了,那么就要了解他,知道有这么一个东西就好. input类型 ...

  2. Unity减少构建安装包的体积(210MB减小到7MB)

    概述 项目简介 由于是公司内做的项目,不方便开源,就只分享优化过程吧. 项目信息 逐日是一个移动端单机小游戏,使用Unity开发,目前已将项目使用的Unity升级到2019.4.14f1c1 (3e5 ...

  3. 2016湖南省赛 A 2016 题解(同余)

    题目链接 题目大意 给出正整数 n 和 m,统计满足以下条件的正整数对 (a, b) 的数量: 1<=a<=n 1<=b<=m a*b%2016=0 题目思路 我本来以为是容斥 ...

  4. Java lambda 分组后多列求和

    主要思路是reducing,可以像sql一样分组后多列求和处理成新对象等: select code,max(name)as name,sum(chengJi)as chengJi,sum(age)as ...

  5. PyQt(Python+Qt)学习随笔: QDoubleSpinBox浮点数字设定部件简介

    专栏:Python基础教程目录 专栏:使用PyQt开发图形界面Python应用 专栏:PyQt入门学习 老猿Python博文目录 老猿学5G博文目录 在<PyQt(Python+Qt)学习随笔: ...

  6. 第15.28节 PyQt(Python+Qt)入门学习:Model/View架构中的便利类QTableWidget详解

    老猿Python博文目录 专栏:使用PyQt开发图形界面Python应用 老猿Python博客地址 一.引言 表格部件为应用程序提供标准的表格显示工具,在表格内可以管理基于行和列的数据项,表格中的最大 ...

  7. Python正则表达式re.match(r"(..)+", "a1b2c3")匹配结果为什么是”c3”?

    在才开始学习正则表达式处理时,老猿对正则表达式:re.match(r"(-)+", "a1b2c3") 返回的匹配结果为"c3"没有理解,学 ...

  8. PyQt(Python+Qt)学习随笔:PyQt帮助文档导入assistant后离线查阅

    在按照<第15.6节 PyQt5安装与配置>完成PyQt5及PyQt5-tools的安装后,发现Qt Designer中的帮助不能使用,报错: 按照<PyQt学习随笔:Qt Desi ...

  9. Fiddle常用命令

    常用命令: 1.启动后点击上方工具栏里有个IE图标的Browse按钮(可以选择Chrome或者Firefox),可以打开系统默认的浏览器,同时也可以确认Fiddler能够抓到浏览器的包. 2.左下角黑 ...

  10. RedHat操作指令第4篇

    top(查看动态进程运行情况) top 是一个更加有用的命令,可以监视系统中不同的进程所使用的资源.它提供实时的系统状态信息. 显示进程的数据包括 PID.进程属主.优先级.%CPU.%memory等 ...