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 中遵循完全相同的规则 。因为方法本质上就是绑定在类上的属性 ,只不过这个属性是可调用的函数

相关推荐
Ulyanov4 小时前
基于 Python 的三维动态导弹攻防演示系统设计与实现:从架构到实战的深度剖析
开发语言·python·qt·架构·雷达电子对抗
Leinwin4 小时前
Claude 四月宕机七次:从一次事故看企业级 AI 部署的容灾设计
后端·python·flask
棉猴4 小时前
Python海龟绘图之绘制文本
javascript·python·html·write·turtle·海龟绘图·输出文本
渣渣盟4 小时前
大数据技术栈全景图:从零到一的入门路线(深度实战版)
大数据·hadoop·python·flink·spark
码农阿豪4 小时前
Python 操作金仓数据库的完全指南(上篇):连接管理与高可用
开发语言·数据库·python
eqwaak04 小时前
4月30号(科技信息差)
python·科技·信息可视化·数据挖掘·数据分析
JaydenAI4 小时前
[MCP在LangChain中的应用-03]在Session构建的上下文中与MCP Server交互
python·langchain·ai编程·ai agent·mcp·fastmcp
X56614 小时前
SQL注入防御技术方案_基于正则表达式的输入清洗
jvm·数据库·python
涛声依旧-底层原理研究所4 小时前
Qwen2.5模型加载与推理实战
人工智能·python