0

0

getattribute 如何避免无限递归的正确防护写法

冰川箭仙

冰川箭仙

发布时间:2026-01-24 18:33:09

|

102人浏览过

|

来源于php中文网

原创

__getattribute__ 容易触发无限递归,因为其内部访问任何属性(如self.__dict__)都会再次调用自身;正确做法是所有属性读取必须显式调用object.__getattribute__(self, name)或super().__getattribute__(name)。

getattribute 如何避免无限递归的正确防护写法

为什么 __getattribute__ 容易触发无限递归

因为每次访问实例的任何属性(包括 self.__dict__self._cache 这类内部变量),都会再次调用 __getattribute__。如果你在方法里直接写 self.xself.__dict__,就等于在递归入口里又踩了一脚递归入口。

正确防护:一律走 object.__getattribute__ 绕过自定义逻辑

所有需要读取属性值的地方,必须显式调用父类实现,不能依赖点号访问。这是唯一可靠方式。

  • 想读原始 __dict__?用 object.__getattribute__(self, '__dict__')
  • 想查某个字段是否已存在?用 object.__getattribute__(self, 'field_name'),别用 hasattr(self, 'field_name')(它内部会触发 __getattribute__
  • 想 fallback 到默认行为?统一用 super().__getattribute__(name),它等价于 object.__getattribute__(self, name)
  • 如果要访问的是描述符(比如 @property),也必须走 super(),否则可能跳过其 __get__ 逻辑

常见错误写法与对应修复

下面这些写法看着自然,但全都会崩:

# ❌ 错误:self.__dict__ 触发递归
def __getattribute__(self, name):
    if name in self.__dict__:  # boom
        return self.__dict__[name]

✅ 正确:绕过自定义逻辑取 dict

def getattribute(self, name): d = object.getattribute(self, 'dict') if name in d: return d[name]

再比如缓存场景:

WeShop唯象
WeShop唯象

WeShop唯象是国内首款AI商拍工具,专注电商产品图片的智能生成。

下载
# ❌ 错误:self._cache 是普通属性,访问即递归
def __getattribute__(self, name):
    cache = self._cache  # boom

✅ 正确:用 super() 或 object.getattribute

def getattribute(self, name): try: cache = super().getattribute('_cache') except AttributeError: cache = {} object.setattr(self, '_cache', cache)

要不要用 __getattr__ 替代?

如果只是想拦截「不存在的属性」,__getattr__ 更安全、更简单,它只在 __getattribute__ 抛出 AttributeError 后才被调用,天然不递归。

  • 适合做 fallback、动态生成、日志兜底
  • 不适合做全局属性拦截(比如统一打日志、权限校验、类型转换),因为它根本看不到已存在的属性
  • 很多场景其实不需要 __getattribute__ —— 先问自己:真要拦住 self.existing_field 吗?还是只拦 self.missing_thing

真正需要 __getattribute__ 的地方不多,一旦用,就必须对每个属性访问都保持警惕,连 self.__class__self.__module__ 都得走 super()

相关专题

更多
C++类型转换方式
C++类型转换方式

本专题整合了C++类型转换相关内容,想了解更多相关内容,请阅读专题下面的文章。

299

2025.07.15

c++ 根号
c++ 根号

本专题整合了c++根号相关教程,阅读专题下面的文章了解更多详细内容。

17

2026.01.23

c++空格相关教程合集
c++空格相关教程合集

本专题整合了c++空格相关教程,阅读专题下面的文章了解更多详细内容。

22

2026.01.23

yy漫画官方登录入口地址合集
yy漫画官方登录入口地址合集

本专题整合了yy漫画入口相关合集,阅读专题下面的文章了解更多详细内容。

91

2026.01.23

漫蛙最新入口地址汇总2026
漫蛙最新入口地址汇总2026

本专题整合了漫蛙最新入口地址大全,阅读专题下面的文章了解更多详细内容。

124

2026.01.23

C++ 高级模板编程与元编程
C++ 高级模板编程与元编程

本专题深入讲解 C++ 中的高级模板编程与元编程技术,涵盖模板特化、SFINAE、模板递归、类型萃取、编译时常量与计算、C++17 的折叠表达式与变长模板参数等。通过多个实际示例,帮助开发者掌握 如何利用 C++ 模板机制编写高效、可扩展的通用代码,并提升代码的灵活性与性能。

14

2026.01.23

php远程文件教程合集
php远程文件教程合集

本专题整合了php远程文件相关教程,阅读专题下面的文章了解更多详细内容。

65

2026.01.22

PHP后端开发相关内容汇总
PHP后端开发相关内容汇总

本专题整合了PHP后端开发相关内容,阅读专题下面的文章了解更多详细内容。

59

2026.01.22

php会话教程合集
php会话教程合集

本专题整合了php会话教程相关合集,阅读专题下面的文章了解更多详细内容。

61

2026.01.22

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PHP自制框架
PHP自制框架

共8课时 | 0.6万人学习

PHP面向对象基础课程(更新中)
PHP面向对象基础课程(更新中)

共12课时 | 0.7万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

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

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