0

0

Python描述符中的递归陷阱:内部属性命名策略解析

霞舞

霞舞

发布时间:2025-09-29 10:27:29

|

677人浏览过

|

来源于php中文网

原创

Python描述符中的递归陷阱:内部属性命名策略解析

本教程深入探讨Python描述符在使用__get__和__set__方法时可能遇到的无限递归问题。核心在于,当描述符内部用于存储值的属性名与描述符在宿主类上的外部属性名相同时,会导致getattr或setattr反复触发描述符自身,从而引发RecursionError。文章将详细解释此机制,并提供通过使用不同内部属性名来有效避免该问题的解决方案及最佳实践。

理解Python描述符

python描述符(descriptor)是实现了特定“描述符协议”方法的对象,这些方法包括__get__、__set__和__delete__。当一个类的实例属性被访问时,如果该属性是一个描述符实例,python解释器就会调用描述符的相应方法来控制属性的访问行为。描述符常用于实现属性验证、缓存、类型转换等高级功能。

__set_name__(self, owner, name)是描述符协议中的一个特殊方法,它在描述符被创建并绑定到宿主类上时被调用。owner参数是宿主类,name参数是描述符在宿主类上的公开名称。这个方法对于描述符内部存储自身状态或关联信息至关重要。

无限递归的根源

当描述符在__get__或__set__方法中尝试通过getattr(instance, self.internal_name)或setattr(instance, self.internal_name, value)来访问或设置实例属性时,如果self.internal_name的值恰好与描述符在宿主类上的公开名称相同,就会导致无限递归。

考虑以下一个有问题的描述符实现:

class ProblematicDescriptor:
    def __set_name__(self, owner, name):
        # 问题所在:内部存储名称与描述符的公开名称相同
        self.internal_name = name 

    def __get__(self, instance, owner):
        if instance is None:
            return self
        print(f"__get__ called for public name '{self.internal_name}'")
        # 此时,getattr(instance, 'some_attribute') 会再次触发描述符的 __get__ 方法
        # 因为 'some_attribute' 正是这个描述符在宿主类上的名称
        return getattr(instance, self.internal_name)

    def __set__(self, instance, value):
        if instance is None:
            return
        print(f"__set__ called for public name '{self.internal_name}'")
        # 同样,setattr(instance, 'some_attribute', value) 会再次触发描述符的 __set__ 方法
        setattr(instance, self.internal_name, value)

class HostClass:
    my_attr = ProblematicDescriptor()

# 尝试访问或设置属性将导致 RecursionError
# host_obj = HostClass()
# host_obj.my_attr = 10  # 尝试设置
# print(host_obj.my_attr) # 尝试获取

当执行host_obj.my_attr = 10时:

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

  1. HostClass.my_attr.__set__(host_obj, 10)被调用。
  2. 在__set__内部,setattr(host_obj, self.internal_name, value)被调用。由于self.internal_name此时为'my_attr',Python解释器会再次尝试设置host_obj.my_attr。
  3. 这会再次触发HostClass.my_attr.__set__(host_obj, 10),形成一个无限循环,直到达到Python的递归深度限制,抛出RecursionError。

__get__方法的调用也遵循相同的逻辑,同样会导致无限递归。

Artflow.ai
Artflow.ai

可以使用AI生成的原始角色、场景、对话,创建动画故事。

下载

解决方案:采用独特的内部属性名

解决此问题的关键在于,在描述符内部用于存储实际值的属性名,必须与描述符在宿主类上的公开名称不同。通过__set_name__方法,我们可以轻松地为内部存储生成一个独特的名称。

通常的做法是在公开名称前加上下划线(_)作为前缀,以表示这是一个内部属性。

class SafeDescriptor:
    def __set_name__(self, owner, name):
        self.public_name = name
        # 解决方案:使用一个不同的内部存储名称,例如加上前缀 '_'
        self.internal_storage_name = f'_{name}' 

    def __get__(self, instance, owner):
        if instance is None:
            return self
        print(f"__get__ called for public name '{self.public_name}', retrieving from '{self.internal_storage_name}'")
        # 此时,getattr(instance, self.internal_storage_name) 会在实例的 __dict__ 中查找
        # 或者沿着 MRO 查找,而不会再次触发描述符协议,从而避免递归。
        return getattr(instance, self.internal_storage_name, None) # 提供默认值以防属性尚未设置

    def __set__(self, instance, value):
        if instance is None:
            return
        print(f"__set__ called for public name '{self.public_name}', storing to '{self.internal_storage_name}'")
        # setattr(instance, self.internal_storage_name, value) 将值存储为实例的一个普通属性
        setattr(instance, self.internal_storage_name, value)

class SafeHostClass:
    my_attr = SafeDescriptor()

# 示例:正确运行
safe_obj = SafeHostClass()
safe_obj.my_attr = 10
print(f"Retrieved value: {safe_obj.my_attr}")

# 验证实例的内部状态
print(f"Instance dictionary: {safe_obj.__dict__}") 
# 输出可能为: Instance dictionary: {'_my_attr': 10}

在这个修正后的实现中,当getattr(instance, self.internal_storage_name)被调用时,Python解释器会查找instance实例中名为_my_attr的普通属性。由于_my_attr不是一个描述符,它不会再次触发SafeDescriptor的__get__方法,而是直接从实例的__dict__中获取值,从而成功打破了递归。setattr也以类似的方式工作。

注意事项与最佳实践

  1. 命名约定: 使用下划线前缀(如_name)是Python社区的通用约定,表示该属性是内部使用的,不应被外部直接访问。这提高了代码的可读性和维护性。
  2. __set_name__的重要性: __set_name__方法是描述符获取其公开名称并生成内部存储名称的关键。它确保了每个描述符实例都能正确地管理其在不同宿主类实例上的值。
  3. getattr与setattr的行为: 理解getattr和setattr在Python中查找属性的机制至关重要。当它们被调用时,Python会首先检查目标对象所属的类是否定义了同名的描述符。如果存在,描述符协议被激活;如果不存在,则在实例的__dict__和类的MRO中查找普通属性。
  4. 直接访问instance.__dict__: 虽然在某些情况下,可以直接通过instance.__dict__[self.internal_storage_name]来访问或设置值,但使用getattr和setattr通常更为推荐,因为它们能更好地处理属性不存在(例如提供默认值)或涉及继承链的情况。但在本递归场景中,关键在于getattr/setattr的目标名称不能是描述符的公开名称。

总结

在Python描述符的实现中,为了避免在__get__和__set__方法中因自身调用而导致的无限递归,核心策略是确保用于存储和检索实际值的内部属性名与描述符在宿主类上的公开名称不同。__set_name__方法提供了获取描述符公开名称的机制,从而允许我们生成一个独特的内部存储名称(例如,通过添加下划线前缀)。遵循这一最佳实践,可以构建健壮且无递归问题的描述符。

相关专题

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

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

716

2023.06.15

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

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

626

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教程的相关文章,大家可以免费体验学习。

1236

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相关的文章、下载、课程内容,供大家免费下载体验。

699

2023.08.11

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

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

62

2025.12.31

热门下载

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

精品课程

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

共4课时 | 0.6万人学习

Django 教程
Django 教程

共28课时 | 2.6万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.0万人学习

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

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