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

相关推荐
luckdewei1 小时前
FastAPI 资产管理系统实战:复杂 ORM 关联、Alembic 迁移与 N+1 查询优化
python
aqi007 小时前
15天学会AI应用开发(八)使用向量数据库实现RAG功能
人工智能·python·大模型·ai编程·ai应用
Csvn8 小时前
`functools.lru_cache` —— 一行代码搞定缓存加速
后端·python
金銀銅鐵1 天前
[Python] 从《千字文》中随机挑选汉字
后端·python
cup111 天前
[技术复盘] Windows Python 打包实战:Nuitka 环境踩坑总结与 CI 自动化构建全指南
python·ai·环境变量·ci·nuitka·skill
aqi001 天前
15天学会AI应用开发(七)有了大模型为什么还要引入RAG
人工智能·python·大模型·ai编程·ai应用
金銀銅鐵1 天前
用 Python 实现 Take-Away 游戏
python·游戏
copyer_xyf1 天前
Agent 流程编排
后端·python·agent