方法解析顺序 / MRO (Method Resolution Order)


关于方法解析顺序(MRO)的详细内容可以参考文末链接,这里主要对 MRO 进行简要的总结说明以及一些练习示例。

经典类和新式类的 MRO


经典类

描述: 一种不能继承的类,如果经典类为父类,其子类调用父类的构造函数时会报错。且不具备 __mro__ 属性。

MRO: Deepth First Search (DFS) / 深度优先搜索

新式类

描述: 新式类是一种为了解决经典类中只能继承不能重写的问题而引入的,新式类默认继承自 object,且子类可以调用父类构造函数,在 Python 之后均为新式类。

MRO: C3算法

下面的代码以 Python2.7 和 Python3.4 分别运行将产生不一样的结果,这是由于在 Python3 中默认使用了新式类,因此采用了 C3 搜索,而 Python2 中默认为经典类,采用 DFS 搜索,因此产生了不同,若将 A 的基类指定为 object 则输出将变为相同。

 """
A
/\
/ \
B C
\ /
\/
D
"""
from distutils.log import warn as printf class A:
def __init__(self):
pass def show(self):
printf("This is A") class B(A): pass class C(A):
def show(self):
printf("This is C") class D(B, C): pass d = D()
d.show()
# Python2 --> This is A
# Python3 --> This is C

DFS / BFS / C3


主要以正常继承和菱形继承为例分别说明三种算法的缺陷和优势。

DFS深度优先搜索

利用深度优先搜索,对于正常继承可以很好的完成搜索,但当出现菱形继承时,则会出现搜索的缺陷,在菱形继承中 MRO 的顺序为 ABDC,若此时 C 重写了 D 的方法,可由于继承顺序的问题,将导致优先搜索到 D 的方法,也就是说,C 的重写无效的,C 对于 D 只能继承,不能重写。

BFS广度优先搜索

广度优先搜索则恰好和深度优先搜索相反,在菱形继承中能够很好的完成搜索,而在普通搜索中却出现了一定缺陷,普通搜索的顺序为ABCDE,但这却违反了单调性原则,即B和C是两个互不相关的父类,在B搜索结束后应该优先搜索D而非C。

C3算法

基于上述的两种算法缺陷,Python2.3之后的新式类中MRO全都采用了C3算法,这种算法解决了DFS的只能继承无法重写的问题和BFS的单调性问题。

Merge List公式


C3 算法的继承模式主要采用了 merge list 公式,首先假设类的线性化 MRO 记为 L[C]=[C1, C2, C3],则称 C1 为 L[C] 的头,其余为尾。若 C 继承自 B1,B2…,则可以根据以下公式计算出 L[C],

L[object] = [object]
L[C(B1…BN)] = [C] + merge(L[B1]…L[BN], [B1]…[BN])

整个计算公式的关键在于 merge,输入为一组列表,输出为一个列表,输出方式为,

1. 检查第一个列表的头元素(如 L[B1] 的头),记为 H;

2. 若 H 未出现在其他列表的尾部,则输出该元素,并删除所有列表中的 H,然后重复步骤 1。若 H 出现在其他列表的尾部,则取下一个列表的头部继续步骤。

3. 重复步骤直至列表为空,则算法结束,若列表不为空且无法输出元素,则说明无法构建继承关系,此时抛出异常。

下面分别举例来构建一个异常的继承关系和正常的继承进行对比,

异常继承

 """
________
/ \
X Y |
\ /\ /
\/ \ /
A B
\ /
\/
C
""" class X(object): pass class Y(object): pass class A(X, Y): pass class B(Y, X): pass class C(A, B): pass print(C.__mro__)

这段代码最终会报错,

Traceback (most recent call last):
File "C:\Users\EKELIKE\Documents\Python Note\3_Program_Structure\3.3_Class\mro_c3_error.py", line 21, in <module>
class C(A, B): pass
TypeError: Cannot create a consistent method resolution
order (MRO) for bases X, Y

原因在于 merge 公式计算到最后无法输出参数,即构建继承失败,计算过程如下,

C3 Algothrim Calculation:
L[object] = [object]
L[X] = [X] + merge(L[object]) = [X, object]
L[Y] = [Y] + merge(L[object]) = [Y, object]
L[A] = [A] + merge(L[X], L[Y], [X], [Y]) = [A, X, Y, object]
L[B] = [B] + merge(L[Y], L[X], [Y], [X]) = [B, Y, X, object]
L[C] = [C] + merge(L[A], L[B], [A], [B])
= [C] + merge([A, X, Y, object], [B, Y, X, object], [A], [B])
= [C, A, B] + merge([X, Y, object], [Y, X, objcet])
--> Raise Error

正常继承

下面给出几个正常继承的示例及其代码,

 # C3 MRO
# ----------------------
"""
Object
/ | \
/ | \
X _ Y Z
\ \/ /
\ /\ /
A B
\/
C
""" class X(object): pass class Y(object): pass class Z(object): pass class A(X, Y): pass class B(X, Z): pass class C(A, B): pass print(C.__mro__)
# ----------------------
 # ----------------------
"""
Object
/ | \
/ | \
D E F
\ / \ /
\ / \/
J K
\ /
L
""" class D(object): pass class E(object): pass class F(object): pass class J(D, E): pass class K(E, F): pass class L(J, K): pass print(L.__mro__)

其中第二种继承的MRO计算公式如下,

C3 Algorithm Calculation:
L[object] = [object]
L[D] = [D] + merge(L[object]) = [D, object]
L[E] = [E] + merge(L[object]) = [E, object]
L[F] = [F] + merge(L[object]) = [F, object]
L[J] = [J] + merge(L[D], L[E], [D], [E]) = [J, D, E, object]
L[K] = [K] + merge(L[E], L[F], [E], [F]) = [K, E, F, object]
L[L] = [L] + merge(L[J], L[K], [J], [K])
= [L, J] + merge([D, E, object], [K, E, F, object], [K])
= [L, J, D] + merge([E, object], [K, E, F, object], [K])
= [L, J, D, K] + merge([E, object], [E, F, object])
= [L, J, D, K, E] + merge([object], [F, object])
= [L, J, D, K, E, F] + merge([object], [object])
= [L, J, D, K, E, F, object]

查看MRO


当然,在使用时我们不必要每次都对类继承顺序进行计算,在 Python 中可以使用以下两种方式来查看一个类的 MRO。

 import inspect

 # Diamond inherit
class A: pass class B(A): pass class C(A): pass class D(B, C): pass print(inspect.getmro(D))
print(D.__mro__)

最终得到输出的结果如下,这一结果即是通过 MRO C3 算法进行计算得出的。

(<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.A'>, <class 'object'>)
(<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.A'>, <class 'object'>)

参考链接


http://ju.outofmemory.cn/entry/137896

http://blog.csdn.net/lis_12/article/details/52859376

Python的程序结构[2] -> 类/Class[2] -> 方法解析顺序 MRO的更多相关文章

  1. Python的方法解析顺序(MRO)[转]

    本文转载自: http://hanjianwei.com/2013/07/25/python-mro/ 对于支持继承的编程语言来说,其方法(属性)可能定义在当前类,也可能来自于基类,所以在方法调用时就 ...

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

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

  3. python 方法解析顺序 mro

    一.概要: mor(Method Resolution Order),即方法解析顺序,是python中用于处理二义性问题的算法 二义性: 1.两个基类,A和B都定义了f()方法,c继承A和B那么C调用 ...

  4. Python的程序结构[2] -> 类/Class[0] -> 类的特殊属性

    类的特殊属性 / Special Property of Class Python 中通过 class 进行类的定义,类可以实例化成实例并利用实例对方法进行调用. 类中还包含的一些共有的特殊属性. 特 ...

  5. Python的程序结构[2] -> 类/Class[1] -> 基类与继承

    基类与继承 / Base Class and Inheritance Class 面向对象的特性使得 Python 中不可避免地需要使用到类和类的继承,类的继承可以使得代码很好的被重用.下面以一些代码 ...

  6. Python的程序结构[2] -> 类/Class[3] -> 内建类与内建函数

    内建类与内建函数的区分 / Distinction of Built-in Type and Function 对于 Python,有许多可以不需要定义或引用就可以使用的函数(类)(参考内建模块),诸 ...

  7. Python的程序结构[2] -> 类/Class[5] -> 内建类 bytes 和 bytearray

    内建类 bytes 和 bytearray / Built-in Type bytes and bytearray 关于内建类 Python的内建类 bytes 主要有以下几点: class byte ...

  8. Python的程序结构[2] -> 类/Class[4] -> 内建类 super

    内建类 super / Built-in Type super 关于内建类 对于 super 可以从官方文档中看到基本介绍,super 接收一个类,以及类或类的实例,最终返回一个代理对象的实例.而 M ...

  9. Python的程序结构[2] -> 类/Class[6] -> 内建类 map

    内建类map / Built-in Type map 关于内建类 map 是一个内建的类,能够返回一个 map 的 obj.map 的第一个参数为一个可执行函数,后续参数均为可迭代对象,map 会分别 ...

随机推荐

  1. 《Cracking the Coding Interview》——第17章:普通题——题目12

    2014-04-29 00:04 题目:给定一个整数数组,找出所有加起来为指定和的数对. 解法1:可以用哈希表保存数组元素,做到O(n)时间的算法. 代码: // 17.12 Given an arr ...

  2. lnmp一键安装环境中nginx开启pathinfo

    问题及原理可参考:http://www.laruence.com/2009/11/13/1138.html 如果是用lnmp脚本一键安装的开发环境,可以通过如下方式开户pathinfo: 1.注释ng ...

  3. mysql:用户管理、索引、视图、函数、存储过程

    #创建一个用户并设置密码,注意IP地址要是登录mysql电脑的IP地址 USE mysql CREATE USER lisi@'192.168.149.1' IDENTIFIED BY "1 ...

  4. winform对图片进行灰度处理

    //图片进行灰度处理 //originalImage为原图像 返回灰度图像 private Bitmap GrayImage(Bitmap originalImage) { ImageAttribut ...

  5. Python全栈工程师(面向对象)

    ParisGabriel                每天坚持手写  一天一篇  决定坚持几年 为了梦想为了信仰    开局一张图 Python人工智能从入门到精通 day16补充: 异常处理 文件 ...

  6. sources-t.list

    deb http://debian.ustc.edu.cn/ubuntu/ trusty main multiverse restricted universe deb http://debian.u ...

  7. [Leetcode/Javascript] 461.Hamming Distance

    [Leetcode/Javascript] 461.Hamming Distance 题目 The Hamming distance between two integers is the numbe ...

  8. Alpha 冲刺

    队名:我头发呢队 组长博客 作业博客 杰(组长) 过去两天完成了哪些任务 查阅Python爬取音源的资料,如 Python3爬虫抓取网易云音乐热评实战 Python爬取高品质QQ音乐(2) 如何爬网易 ...

  9. 在LinkedIn的 Kafka 生态系统

    在LinkedIn的 Kafka 生态系统 Apache Kafka是一个高度可扩展的消息传递系统,作为LinkedIn的中央数据管道起着至关重要的作用. Kafka 是在2010年在LinkedIn ...

  10. Citrix NetScaler HA(高可用性)解析

    Citrix NetScaler HA(高可用性)解析 来源 https://www.iyunv.com/thread-172259-1-1.html 1.1     NetScaler高可用概述 我 ...