方法解析顺序 / 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. APP开发手记01(app与web的困惑)

    文章链接:http://quke.org/post/app-dev-fragment.html (转载时请注明本文出处及文章链接) 最近在用博客园的wcf服务做博客园的android和ios的app, ...

  2. 跟着学!教你开发第一个Java程序

    今天我们的目标是开发人生中的第一个Java程序,虽然可能会很简单,但是这小小的一步却是跨入IT行业的一大步!下面我们来一起来仔细的了解开发的流程. 准备工作 1,作为一名准程序猿自备一台电脑那是必不可 ...

  3. java中继承的概念

    继承是类的三大特性之一,是java中实现代码重用的重要手段之一.       java中只支持单继承,即每个类只能有一个父类.       继承表达的是is a的关系,或者说一种特殊和一般的关系.   ...

  4. katalon系列一:初识Katalon Studio自动化测试工具

    最近准备把公司的系统搞上UI自动化,先是自己用Python+selenium+pytest写了一个框架,开始写case的时候发现效率极其慢.原因为: (1)开发为提高前端响应时间,使用前端路由技术,一 ...

  5. GLIBCXX3.4.21 not find

    在执行世界杯的二进制代码和安装keepaway中会遇到GLIBCXX3.4.21 not find的问题,其解决办法就是升级安装GCC. 一.首先查看当前gcc版本 $ strings /usr/li ...

  6. mysql Access denied for user 'root'@'localhost'问题解决

    问题描述: 系统安装mysql的过程中,没有提示配置用户名和密码相关的信息,安装完毕后,登录报错. 表现现象为: mysql -u root -p [输入root密码] 界面提示: ERROR 169 ...

  7. [oldboy-django][2深入django]form表单clean_xx, clean完成数据验证+ form错误信息

    form后台生成form里面的Input标签,以及设置Input的属性 # 需求 后台生成form里面的input标签,并设置input标签的属性, class RegisterForm(Form): ...

  8. HDU 4665 Unshuffle DFS找一个可行解

    每层找一对相等的整数,分别放在两个不同的串中. 参考了学弟的解法,果断觉得自己老了…… #include <cstdio> #include <cstring> #includ ...

  9. mysql安装目录、配置文件存放位置

    linux系统下,如何知道mysql使用的配置文件到底是哪个呢?linux自带的mysql的安装目录又是什么呢?数据存放在什么目录下? 1.linux系统自带的mysql,其安装目录及数据目录查看方法 ...

  10. PHP命名空间与use

    当在一个大型项目很多程序员书写模板时,最怕出现的问题就是命名,如果一个PHP脚本出现了同名的类或者方法,就会报错(fatal error),使用命名空间可以 解决这个问题 知识点: 命名空间names ...