0

0

Python描述符中的递归陷阱:内部属性名管理最佳实践

花韻仙語

花韻仙語

发布时间:2025-09-29 12:13:01

|

661人浏览过

|

来源于php中文网

原创

Python描述符中的递归陷阱:内部属性名管理最佳实践

本文深入探讨了Python描述符在使用__get__和__set__方法时可能遇到的RecursionError。当描述符的内部存储名称与它在宿主类上绑定的属性名称相同时,getattr和setattr操作会导致无限递归。文章将详细解释这一机制,并提供通过使用不同的内部属性名来避免递归的解决方案和最佳实践。

理解Python描述符与属性访问

python描述符是实现特定协议的对象,它们通过定义__get__、__set__和__delete__方法来控制类属性的访问行为。当一个描述符实例被放置在类的字典中时,对该类实例上相应属性的访问(获取、设置、删除)将被委托给描述符的相应方法。

一个典型的描述符模式是,它在宿主实例上存储实际的数据。为了实现这一点,描述符需要一个名称来在宿主实例的__dict__中查找或设置值。这个名称通常在描述符的__set_name__方法中被初始化。

递归陷阱的产生

考虑以下一个尝试实现简单数据存储的描述符:

class MyDescriptor:
    def __init__(self, default_value=None):
        self.default_value = default_value
        self.internal_name = None # 将在__set_name__中设置

    def __set_name__(self, owner, name):
        # 错误示范:将内部名称设置为与描述符绑定的外部名称相同
        self.internal_name = name 

    def __get__(self, instance, owner):
        if instance is None:
            return self # 当通过类访问时返回描述符本身
        # 错误示范:这里会引发递归
        return getattr(instance, self.internal_name)

    def __set__(self, instance, value):
        if instance is None:
            raise AttributeError("Cannot set attribute on class directly.")
        # 错误示范:这里也会引发递归
        setattr(instance, self.internal_name, value)

class MyClass:
    # 描述符被绑定到 'data' 这个名称
    data = MyDescriptor(default_value=0)

# 尝试使用 MyClass
# instance = MyClass()
# print(instance.data) # 这会引发 RecursionError

当执行print(instance.data)时,Python会尝试获取instance.data的值。由于data是一个描述符,Python会调用MyDescriptor实例的__get__(instance, MyClass)方法。

在__get__方法内部,我们有return getattr(instance, self.internal_name)。此时,self.internal_name的值是'data'(因为它是在__set_name__中被设置为name参数的值)。因此,这个表达式等同于getattr(instance, 'data')。

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

问题在于,getattr(instance, 'data')会再次触发对instance.data的访问,而instance.data又是一个描述符,于是Python会再次调用MyDescriptor实例的__get__方法。这就形成了一个无限递归循环:__get__调用getattr,getattr又调用__get__,直到达到Python的最大递归深度限制,抛出RecursionError。

__set__方法中的setattr(instance, self.internal_name, value)也会遇到同样的问题,因为它同样会重新触发对描述符的调用。

稿定AI绘图
稿定AI绘图

稿定推出的AI绘画工具

下载

解决方案:使用不同的内部属性名

解决这个问题的关键在于,描述符内部用于存储实际值的属性名,必须与描述符在宿主类上被绑定的外部属性名不同。通常,我们会选择在内部属性名前加上一个下划线(_)作为约定。

修改后的__set_name__方法如下:

class MyDescriptor:
    def __init__(self, default_value=None):
        self.default_value = default_value
        self.internal_name = None

    def __set_name__(self, owner, name):
        # 修正:将内部名称设置为与描述符绑定的外部名称不同的值
        self.internal_name = f'_{name}' 

    def __get__(self, instance, owner):
        if instance is None:
            return self
        # 修正:现在 getattr(instance, self.internal_name) 将直接访问实例的 __dict__
        # 而不会再次触发描述符的 __get__ 方法
        if hasattr(instance, self.internal_name):
            return getattr(instance, self.internal_name)
        return self.default_value # 如果实例上还没有这个属性,返回默认值

    def __set__(self, instance, value):
        if instance is None:
            raise AttributeError("Cannot set attribute on class directly.")
        # 修正:setattr(instance, self.internal_name, value) 将直接在实例的 __dict__ 中设置值
        setattr(instance, self.internal_name, value)

class MyClass:
    data = MyDescriptor(default_value=0)
    name = MyDescriptor(default_value="Unnamed")

# 完整示例
if __name__ == "__main__":
    print("--- 使用修正后的描述符 ---")
    instance1 = MyClass()
    print(f"实例1的默认data: {instance1.data}") # 输出: 实例1的默认data: 0
    print(f"实例1的默认name: {instance1.name}") # 输出: 实例1的默认name: Unnamed

    instance1.data = 100
    instance1.name = "Alice"
    print(f"实例1设置后的data: {instance1.data}") # 输出: 实例1设置后的data: 100
    print(f"实例1设置后的name: {instance1.name}") # 输出: 实例1设置后的name: Alice

    instance2 = MyClass()
    print(f"实例2的默认data: {instance2.data}") # 输出: 实例2的默认data: 0
    print(f"实例2的默认name: {instance2.name}") # 输出: 实例2的默认name: Unnamed

    # 验证不同实例的数据独立性
    instance2.data = 200
    print(f"实例1的data (未变): {instance1.data}") # 输出: 实例1的data (未变): 100
    print(f"实例2的data (已变): {instance2.data}") # 输出: 实例2的data (已变): 200

    # 尝试直接访问内部属性(不推荐,但可用于理解)
    # print(instance1._data) # AttributeError: '_data'
    # 解释:_data 是一个常规属性,但它存在于实例的 __dict__ 中,
    # 默认情况下,如果描述符没有定义,直接访问 _data 是可以的。
    # 但由于描述符控制了 'data' 的访问,我们通常不直接访问 _data。
    # 这里的关键是 getattr(instance, '_data') 不会触发描述符。
    print(f"直接访问实例内部存储的data: {getattr(instance1, '_data')}") # 输出: 直接访问实例内部存储的data: 100

通过将self.internal_name设置为f'_{name}',例如当描述符绑定到data时,内部存储的名称变为_data。现在:

  • getattr(instance, self.internal_name) 变成了 getattr(instance, '_data')。
  • setattr(instance, self.internal_name, value) 变成了 setattr(instance, '_data', value)。

由于_data是一个在宿主实例instance上直接存储的普通属性,而不是一个描述符,因此getattr(instance, '_data')和setattr(instance, '_data', value)将直接在instance的__dict__中查找或设置名为_data的属性,而不会再次触发MyDescriptor的__get__或__set__方法。这样就成功打破了递归循环。

注意事项与最佳实践

  1. 命名约定:使用下划线前缀(如_name)是Python中表示“内部使用”的常见约定。这有助于清晰地区分描述符的外部公共名称和其内部存储的实际属性名称。
  2. __set_name__的重要性:__set_name__方法是Python 3.6+引入的,它使得描述符能够知道它被绑定到类上的名称。这是实现这种内部名称管理模式的关键。
  3. instance is None检查:在__get__方法中,当通过类而不是实例访问描述符时(例如MyClass.data),instance参数将是None。在这种情况下,通常应该返回描述符实例本身,而不是尝试获取实例上的值。__set__方法通常不允许直接在类上设置值,因此在instance is None时抛出AttributeError是合理的。
  4. 初始化与默认值:在__get__方法中,当首次访问一个尚未设置的属性时,需要确保返回一个合理的值(如默认值)。通过hasattr(instance, self.internal_name)可以检查实例上是否已经存在该内部属性。

总结

Python描述符是一个强大而灵活的机制,用于定制属性访问。然而,在实现自定义__get__和__set__方法时,必须特别注意避免无限递归。核心原则是:描述符内部用于存储和检索实际值的属性名,必须与描述符在宿主类上绑定的外部属性名不同。通过在__set_name__中生成一个带有下划线前缀的内部名称,我们可以确保getattr和setattr操作直接作用于实例的__dict__,从而有效地防止递归,并使描述符按预期工作。理解并遵循这一模式,是编写健壮Python描述符的关键。

相关专题

更多
python开发工具
python开发工具

php中文网为大家提供各种python开发工具,好的开发工具,可帮助开发者攻克编程学习中的基础障碍,理解每一行源代码在程序执行时在计算机中的过程。php中文网还为大家带来python相关课程以及相关文章等内容,供大家免费下载使用。

715

2023.06.15

python打包成可执行文件
python打包成可执行文件

本专题为大家带来python打包成可执行文件相关的文章,大家可以免费的下载体验。

625

2023.07.20

python能做什么
python能做什么

python能做的有:可用于开发基于控制台的应用程序、多媒体部分开发、用于开发基于Web的应用程序、使用python处理数据、系统编程等等。本专题为大家提供python相关的各种文章、以及下载和课程。

739

2023.07.25

format在python中的用法
format在python中的用法

Python中的format是一种字符串格式化方法,用于将变量或值插入到字符串中的占位符位置。通过format方法,我们可以动态地构建字符串,使其包含不同值。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

617

2023.07.31

python教程
python教程

Python已成为一门网红语言,即使是在非编程开发者当中,也掀起了一股学习的热潮。本专题为大家带来python教程的相关文章,大家可以免费体验学习。

1235

2023.08.03

python环境变量的配置
python环境变量的配置

Python是一种流行的编程语言,被广泛用于软件开发、数据分析和科学计算等领域。在安装Python之后,我们需要配置环境变量,以便在任何位置都能够访问Python的可执行文件。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

547

2023.08.04

python eval
python eval

eval函数是Python中一个非常强大的函数,它可以将字符串作为Python代码进行执行,实现动态编程的效果。然而,由于其潜在的安全风险和性能问题,需要谨慎使用。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

575

2023.08.04

scratch和python区别
scratch和python区别

scratch和python的区别:1、scratch是一种专为初学者设计的图形化编程语言,python是一种文本编程语言;2、scratch使用的是基于积木的编程语法,python采用更加传统的文本编程语法等等。本专题为大家提供scratch和python相关的文章、下载、课程内容,供大家免费下载体验。

698

2023.08.11

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

7

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 0.6万人学习

Django 教程
Django 教程

共28课时 | 2.6万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.0万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号