一. python多继承

  在前面的学习过程中,我们已经知道了python中类与类之间可以有继承关系,当出现x是一种y的时候就可以使用继承关系.即'is-a'关系,在继承关系中子类自动拥有父类中除了私有属性外的其他所有内容,python支持多继承,一个类可以拥有多个父类

class ShenXian: # 神仙
def fei(self):
print("神仙都会飞") class Monkey: # 猴
def chitao(self):
print("猴子喜欢吃桃子")
class SunWukong(ShenXian, Monkey): # 孙悟空是神仙, 同时也是一只猴
pass sxz = SunWukong() # 孙悟空
sxz.chitao() # 会吃桃子
sxz.fei() # 会飞

  此时孙悟空是一只猴子, 同时也是一个神仙,那孙悟空继承了这两个类,孙悟空自然就可以执行这两个类中的方法.

  多继承用起来简单,也很好理解,但是多继承中, 存在着这样一个问题,当两个父类中出现了重名方法的时候,这时该怎么办呢? 这时就涉及到如何查找父类方法的这么一个问题,即MRO(method resolution order) 问题,在python中这是一个很复杂的问题,因为在不同的python版本中使用的是不同的算法来完成MRO的,首先我们目前能见到的有两个版本:

    python2

      在python2中存在两种类:

      一个叫经典类,在python2.2之前一直使用的是经典类,经典类在基类的根如果什么都不写,表示继承xxx

      一个叫新式类,在python2.2之后出现了新式类,新式类的特点是基类的根是object

    python3

      python3中使用的都是新式类,如果基类谁都不继承,那这个类会默认继承object

二.经典类的MRO

  虽然在python3中已经不存在经典类了,但是经典类的MRO最好还是学一学,这是一种树形结构遍历的一个最直接的案例,在python的继承体系中,我们可以把类与类继承关系化成一个树形结构的图. 来, 上代码:

class A:
pass
class B(A):
pass
class C(A):
pass
class D(B, C):
pass
class E:
pass
class F(D, E):
pass
class G(F, D):
pass
class H:
pass
class Foo(H, G):
pass

  对付这样的MRO,很简单,画图即可:

  继承关系图已经有了,那如何进行查找呢? 记住一个原则,在经典类中采用的是深度优先遍历方案,什么是深度优先,就是一条路走到头,然后再回来,继续找下一个. 比如,有一个快递员,去给每家每户送鸡蛋.

 

  图中每个圈都是准备要送鸡蛋的住址,箭头和黑线表示线路,那送鸡蛋的顺序告诉你入口在最下面R,并且必须从左往右送,那怎么送呢?

  如图,肯定是按照123456这样的顺序来送,那这样的顺序就叫深度优先遍历, 而如果是142356呢? 这种被称为广度优先遍历. 好了,深度优先就说这么多,那么上面那个图怎么找的呢? MRO是什么呢? 很简单,记住,从头开始,从左往右,一条路跑到头, 然后回头,继续一条路跑到头,就是经典类的MRO算法.

  类的MRO: Foo-> H -> G -> F -> D -> B -> A -> C -> E,你猜对了么?

三.新式类的MRO

  python中的新式类的MRO是采用的C3算法来完成的.

  c3算法很简单,就看你的代码就够了,不需要去画图,而且画图也看不出来什么,不过如果写得多了是可以从图上总结出一些规律来的,先看代码:

class A:
pass
class B(A):
pass
class C(A):
pass
class D(B, C):
pass
class E(C, A):
pass
class F(D, E):
pass
class G(E):
pass
class H(G, F):
pass

  首先,我们要确定从H开始找,也就是说,创建的是H的对象.

  如果从H找,那找到H+H的父类的C3, 我们设C3算法是L(x) , 即给出x类,找到x的MRO

    L(H) = H + L(G) + L(F) + GF

  继续从代码中找G和F的父类往里面带

    L(G) = G + L(E) + E

    L(F) = F + L(D)+ L(E) + DE

  继续找E 和 D

    L(E) = E + L(C) + L(A) + CA

    L(D) = D + L(B) + L(C) + BC

  继续找B和C

    L(B) = B + L(A) + A

    L(C) = C + L(A) + A

  最后就剩下一个A了,也就不用再找了,接下来,把L(A) 往里带,再推回去,但要记住,这里的+ 表示的是merge,merge的原则是用每个元组的头一项和后面元组的除头一项外的其他元素进行比较, 看是否存在,如果存在,就从下一个元组的头一项继续找,如果找不到,就拿出来,作为merge的结果的一项,以此类推,直到元组之间的元素都相同,也就不用再找了.

    L(B) =(B,) + (A,) + (A) -> (B, A)

    L(C) =(C,) + (A,) + (A) -> (C, A)

  继续带

    L(E) = (E,) + (C, A) + (A) + (C,A) -> E, C, A

    L(D) = (D,) + (B, A) + (C, A) + (B, C) -> D, B, C, A

  继续带

    L(G) = (G,) + (E, C, A) + (E) -> G, E, C, A

    L(F) = (F,) + (D, B, C, A) + (E, C, A) + (D, E)-> F, D, B, E, C, A

  加油, 最后了

    L(H) = (H, ) + (G, E, C, A) + ( F, D, B, E, C, A) + (G, F) -> H, G, F, D, B, E, C, A

  算完了,最终结果 HGFDBECA. 那这个算完了,如何验证呢? 其实python早就给你准备好了,我们可以使用类名.__mro__获取到类的MRO信息.

print(H.__mro__)
结果:
(<class '__main__.H'>, <class '__main__.G'>, <class '__main__.F'>, <class
'__main__.D'>, <class '__main__.B'>, <class '__main__.E'>, <class
'__main__.C'>,<class '__main__.A'>, <class 'object'>)

  结果OK,那既然python提供了,为什么我们还要如此麻烦的计算MRO呢? 因为笔试.......你在笔试的时候, 是没有电脑的,所以这个算法要知道,并且简单的计算要会,真是项目开发的时候很少有人这么去写代码.

  这个说完了,那C3到底怎么看更容易呢? 其实很简单,C3是把我们多个类产生的共同继承留到最后去找,所以,我们也可以从图上来看到相关的规律,这个要大家自己多写多画图就能感觉到了,但是如果没有所谓的共同继承关系,那几乎就当成是深度遍历就可以了.

四.  super是什么鬼?

  super()可以帮我们执行MRO中下一个父类的方法,通常super()有两个使用的地方:

    1. 可以访问父类的构造方法

    2. 当子类方法想调用父类(MRO)中的方法

  我们先看第一种:

class Foo:
def __init__(self, a, b, c):
self.a = a
self.b = b
self.c = c class Bar(Foo):
def __init__(self, a, b, c, d):
super().__init__(a, b, c) # 访问父类的构方法
self.d = d b = Bar(1, 2, 3, 4)
print(b.__dict__)
结果:
{'a': 1, 'b': 2, 'c': 3, 'd': 4}

  这样就方便了子类,不需要写那么多了,直接用父类的构造帮我们完成一部分代码

  第二种:

class Foo:
def func1(self):
super().func1() # 此时找的是MRO顺序中下一个类的func1()方法
print("我的老家. 就住在这个屯") class Bar:
def func1(self):
print("你的老家. 不在这个屯") class Ku(Foo, Bar):
def func1(self):
super().func1() # 此时super找的是Foo
print("他的老家. 不知道在哪个屯") k = Ku() # 先看MRO . KU, FOO, BAR object
k.func1() k2 = Foo() # 此时的MRO. Foo object
k2.func1() # 报错

  最后给大家留一道题

class Init(object):
def __init__(self, v):
print("init")
self.val = v class Add2(Init):
def __init__(self, val):
print("Add2")
super(Add2, self).__init__(val)
print(self.val)
self.val += 2 class Mult(Init):
def __init__(self, val):
print("Mult")
super(Mult, self).__init__(val)
self.val *= 5 class HaHa(Init):
def __init__(self, val):
print("哈哈")
super(HaHa, self).__init__(val)
self.val /= 5 class Pro(Add2,Mult,HaHa): #
pass class Incr(Pro): def __init__(self, val):
super(Incr, self).__init__(val)
self.val+= 1 # Incr Pro Add2 Mult HaHa Init p = Incr(5)
print(p.val) c = Add2(2)
print(c.val)

  提示, 先算MRO , 然后看清楚self是谁.

结论: 不管super()写在哪儿,在哪儿执行,一定先找到MRO列表,根据MRO列表的顺序往下找,否则一切都是错的

Python-MRO和C3算法的更多相关文章

  1. python MRO及c3算法

    1. 了解python2和python3类的区别 python2在2.3之前使用的是经典类, 2.3之后, 使用的是新式类 2. 经典类的MRO 树形结构的深度优先遍历 -> 树形结构遍历 cl ...

  2. Python的多继承问题-MRO和C3算法

    大部分内容转载自C3 线性化算法与 MRO 理解Python中的多继承 Python 中的方法解析顺序(Method Resolution Order, MRO)定义了多继承存在时 Python 解释 ...

  3. day21 MRO和C3算法

    核能来袭 --MRO和C3算法 1. python的多继承 2.python经典类的MRO 3.python新式类的MRO, C3算法 4.super 是什么鬼? 一.python的多继承 在前面的学 ...

  4. Python之MRO及其C3算法

    [<class '__main__.B'>, <class '__main__.A'>, <class 'object'>] (<class '__main_ ...

  5. python全栈开发day103-python垃圾回收机制、mro和c3算法解析、跨域jsonp\CORS、Content-Type组件

    Python垃圾回收 -- 引用计数 -- Python为每个对象维护一个引用计数 -- 当引用计数为0的 代表这个对象为垃圾 -- 标记清除 -- 解决孤立的循环引用 -- 标记根节点和可达对象 - ...

  6. python之路--MRO和C3算法

    一 . MRO(method resolution order) 多继承的一种方法,一种查找的顺序 在python3 里面是一种新类式MRO 需要用都的是C3算法 class A: pass clas ...

  7. python之MRO和C3算法

    python2类和python3类的区别pyhon2中才分新式类与经典类,python3中统一都是新式类Python 2.x中默认都是经典类,只有显式继承了object才是新式类python 3.x中 ...

  8. python中的MRO和C3算法

    一. 经典类和新式类 1.python多继承 在继承关系中,python子类自动用友父类中除了私有属性外的其他所有内容.python支持多继承.一个类可以拥有多个父类 2.python2和python ...

  9. python摸爬滚打之day20--多继承,MRO和C3算法

    1.新式类和经典类 在python2.2之前, 基类如果不写(), 则表示为经典类; 在python2.2之后, 经典类不复存在, 只存在新式类. 如果基类谁都不继承的话, 则默认继承object. ...

  10. Python多重继承顺序---C3算法

    什么是多重继承C3算法 MRO即 method resolution order (方法解释顺序),主要用于在多继承时判断属性的路径(来自于哪个类). 在python2.2版本中,算法基本思想是根据每 ...

随机推荐

  1. Kettle入门及性能优化FAQ

    1.安装 配置Java环境 Java环境配置问题 java_home:D:\Program Files\Java\jdk1.7.0_25(安装jdk路径) classpath:.;%java_home ...

  2. 理解webpack中的publicPath

    outPut中的publicPath 默认值: 空字符串. publicPath是非常有必要配置的,他是项目中引入静态资源(js.css)时的基础路径. 例如: outPut.publicPath = ...

  3. ipython 导入模块 非法ascii字符问题

    SyntaxError: Non-ASCII character '\xe2‘ ipython  使用了 python2    加入 #coding=utf-8 ipython3 使用 python3

  4. vue中vueRouter使用

    首先需要安装依赖:

  5. 06. 用css实现三角形

    用css实现三角形   <!DOCTYPE html> <html lang="en"> <head> <meta charset=&qu ...

  6. 洛谷 P3201 [HNOI2009]梦幻布丁(启发式合并)

    题面 luogu 题解 什么是启发式合并? 小的合并到大的上面 复杂度\(O(nlogn)\) 这题颜色的修改,即是两个序列的合并 考虑记录每个序列的\(size\) 小的合并到大的 存序列用链表 但 ...

  7. Windows搭建Go语言环境·

    对于Windows用户,Go语言提供两种安装方式(源码安装除外): .MSI安装:程序会自动配置你的安装 .ZIP安装:需要你手动设置一些环境变量 一.MSI安装 1.下载安装包(根据操作系统选择相应 ...

  8. CentOS&.NET Core初试-3-Nginx的安装和配置

    系列目录 CentOS的安装和网卡的配置 安装.NET Core SDK和发布网站 Nginx的安装和配置 安装守护服务(Supervisor) Nginx简介   Nginx是一个免费的,开源的,高 ...

  9. oracle使用已有vid快速新建虚拟机

    有时候需新建虚拟机,但是每一次新建都花费很多时间,这里在oracle中快速新建虚拟机,只需要复制已有的vdi,然后修改uuid即可   windows下在cmd下输入命令 cd C:\Program ...

  10. SpringCloud---API网关服务---Spring Cloud Zuul

    1.概述 1.1 微服务架构出现的问题   及  解决: 1.1.1 前言 每个微服务应用都提供对外的Restful API服务,它通过F5.Nginx等网络设备或工具软件实现对各个微服务的路由与负载 ...