Python 属性访问的 MRO 规则

MRO(Method Resolution Order,方法解析顺序)是 Python 中多继承环境下 查找属性和方法的确定性顺序 。Python 使用 C3 线性化算法来计算 MRO。

一、MRO 的规则

1. 查找顺序

复制代码
class A:
    value = "A"

class B(A):
    value = "B"

class C(B):
    pass

c = C()
print(c.value)  # 输出: B

查找路径

复制代码
C → B → A → object

2. 查找规则

规则 说明
从左到右 多继承时,先查第一个父类,再查第二个
从下到上 先查子类,再查父类
子类优先 子类会覆盖父类的同名属性
C3 线性化 保证单调性(不会出现不一致的顺序)

二、MRO 的确定方法

1. 使用 __mro__ 查看

复制代码
class GrandParent:
    pass

class Parent1(GrandParent):
    pass

class Parent2(GrandParent):
    pass

class Child(Parent1, Parent2):
    pass

print(Child.__mro__)

输出

复制代码
(<class '__main__.Child'>, 
 <class '__main__.Parent1'>, 
 <class '__main__.Parent2'>, 
 <class '__main__.GrandParent'>, 
 <class 'object'>)

2. 使用 mro() 方法

复制代码
print(Child.mro())  # 与 __mro__ 相同,但返回列表

3. 使用 help(ClassName) 查看

复制代码
help(Child)
# 会显示 Method resolution order 部分

三、C3 线性化算法原理

1. 算法核心公式

复制代码
L[C(B1...BN)] = C + merge(L[B1], L[B2], ..., L[BN], B1...BN)

其中:

  • L[C] 表示类 C 的线性化结果

  • merge 是合并操作,保证顺序

2. 手动计算示例

复制代码
class O: pass
class A(O): pass
class B(O): pass
class C(A, B): pass

计算过程

复制代码
L[O] = O
L[A] = A + merge(L[O], O) = A + merge(O, O) = A, O
L[B] = B, O
L[C] = C + merge(L[A], L[B], A, B)
     = C + merge([A,O], [B,O], [A,B])
     = C + [A] + merge([O], [B,O], [B])
     = C + [A,B] + merge([O], [O])
     = C + [A,B,O]

结果C → A → B → O

四、多继承的 MRO 示例

菱形继承

复制代码
class A:
    value = "A"

class B(A):
    value = "B"

class C(A):
    value = "C"

class D(B, C):
    pass

print(D.__mro__)
print(D.value)  # 输出: B

MRO

复制代码
D → B → C → A → object

查找 value

  1. D:没有 → 2. B:有 "B" → 停止

五、属性访问的查找流程

代码验证

复制代码
class Descriptor:
    def __get__(self, instance, owner):
        return "描述符的值"

class Parent:
    class_attr = "父类的类属性"
    desc = Descriptor()

class Child(Parent):
    pass

c = Child()

# 1. 实例属性
c.instance_attr = "实例属性"
print(c.instance_attr)  # 实例属性

# 2. 类属性(按 MRO)
print(c.class_attr)     # 父类的类属性

# 3. 描述符(特殊处理)
print(c.desc)           # 描述符的值

方法的查找和属性的查找在 Python 中遵循完全相同的规则 。因为方法本质上就是绑定在类上的属性 ,只不过这个属性是可调用的函数

相关推荐
我星期八休息11 小时前
Linux系统编程—库制作与原理
linux·运维·服务器·数据结构·人工智能·python·散列表
Cloud_Shy61811 小时前
Python 数据分析基础入门:《Excel Python:飞速搞定数据分析与处理》学习笔记系列(第十二章 用户定义函数 上篇)
python·数据分析·excel·pandas
BU摆烂会噶11 小时前
【LangGraph】House_Agent 实战(四):预定流程 —— 中断与人工干预
android·人工智能·python·langchain
AI玫瑰助手11 小时前
Python运算符:比较运算符(等于不等等于大于小于)与返回值
android·开发语言·python
GIOTTO情11 小时前
Infoseek舆情处置系统的技术实现与落地实践
python
new_dev12 小时前
Python实现Android自动化打包工具:加固、签名、多渠道一键完成
android·python·自动化
天天进步201512 小时前
从零打造 Python 全栈项目:智能教学辅助系统
开发语言·人工智能·python
带带弟弟学爬虫__12 小时前
dyAPP数据采集-个人主页、发布、搜索、评论
服务器·python·算法·flutter·java-ee·django
还是鼠鼠12 小时前
AI掘金头条新闻系统 (Toutiao News)-相关推荐
后端·python·mysql·fastapi·web
数智工坊12 小时前
PyCharm 运行 Python 脚本总自动进 Test 模式?附 RT-DETRv2 依赖缺失终极排坑
开发语言·ide·人工智能·python·pycharm