Python类属性访问流程_查找顺序解析【教程】

8次阅读

Python 属性查找顺序为:先实例__dict__,再按 MRO 搜索类及其父类,描述符会改变优先级,特殊方法隐式查找遵循 MRO 但跳过不可调用实例属性,__getattr__是最后兜底。

Python 类属性访问流程_查找顺序解析【教程】

Python 中访问类属性时,解释器会按特定顺序搜索属性,这个顺序直接影响代码行为,尤其在继承和实例化场景下容易出错。理解查找流程是写出可维护、无歧义代码的基础。

实例属性优先于类属性

当通过实例访问属性(如 obj.attr)时,Python 首先检查该实例的 __dict__ 中是否存在该属性。如果存在,直接返回,不会继续向上查找。

  • 即使同名类属性已定义,只要实例自身有该属性,就屏蔽类属性
  • 赋值操作 obj.attr = value 默认创建或修改实例属性,而非修改类属性
  • 想显式修改类属性,需用 ClassName.attr = valuetype(obj).attr = value

类属性查找遵循 MRO(方法解析顺序)

当实例没有该属性,或直接通过类访问(如 MyClass.attr),Python 按 MRO 顺序在类及其父类中查找。MRO 由 C3 线性化算法确定,可通过 ClassName.__mro__ 查看。

  • 查找从左到右,一旦在某个类中找到属性,立即返回,不再继续
  • 多继承下,同名属性出现在靠前的父类中会覆盖靠后的定义
  • 使用 super() 调用时,也严格遵循 MRO 顺序向后查找

描述符会改变默认查找逻辑

如果查找到的属性是一个实现了 __get____set____delete__ 的描述符,Python 会触发对应协议方法,而不是直接返回属性值。这是 property、classmethod、staticmethod 等机制的底层原理。

立即学习 Python 免费学习笔记(深入)”;

  • 数据描述符(含 __set__)优先级高于实例属性
  • 非数据描述符(仅含 __get__)优先级低于实例属性
  • 常见陷阱:用 @property 定义只读属性后,又在实例上赋值 obj.attr = x,此时若未定义 __set__,赋值会成功但创建实例属性,掩盖 property

特殊属性和内置方法有隐式查找规则

某些操作会隐式触发特定名称的属性查找,例如 len(obj) 查找 obj.__len__obj + other 查找 obj.__add__。这类查找同样遵守 MRO,但跳过实例字典中的同名非可调用对象。

  • 即使实例 __dict__ 中有 ‘__len__’ 键,只要其值不可调用,len() 仍会继续沿 MRO 查找
  • 自定义 __getattr__ 是最后兜底机制,仅在常规查找全部失败后才被调用
  • __getattribute__ 则拦截所有属性访问,慎用,避免无限递归

不复杂但容易忽略。掌握这个流程,能快速定位属性未生效、意外覆盖、继承冲突等问题。

星耀云
版权声明:本站原创文章,由 星耀云 2026-01-02发表,共计1113字。
转载说明:转载本网站任何内容,请按照转载方式正确书写本站原文地址。本站提供的一切软件、教程和内容信息仅限用于学习和研究目的;不得将上述内容用于商业或者非法用途,否则,一切后果请用户自负。本站信息来自网络,版权争议与本站无关。
text=ZqhQzanResources