__getattribute__在每次访问属性时调用,__getattr__在属性不存在时触发,__setattr__在设置属性时执行;三者用于控制属性访问与赋值,需避免直接访问属性导致无限递归,应使用super()调用父类方法。

Python中通过魔法函数可以灵活控制类的属性访问和管理。最常用的三种属性管理相关的魔法函数是 __getattribute__、__getattr__ 和 __setattr__。它们在属性读取和赋值时自动触发,掌握它们的区别和使用场景对深入理解Python对象机制非常重要。
__getattribute__(self, name)
这个方法在**每次访问任意属性时都会被调用**,无论属性是否存在。它是属性访问的第一道关口。
注意:由于它总是被调用,如果在里面使用 self.attr 这样的方式访问属性,会再次触发 __getattribute__,容易造成无限递归。常见用途包括:
- 拦截所有属性访问进行日志记录或权限检查
- 实现通用的属性代理或装饰
示例:
立即学习“Python免费学习笔记(深入)”;
def __getattribute__(self, name):
print(f"访问属性: {name}")
return super().__getattribute__(name)
__getattr__(self, name)
只有在属性**不存在**于对象中时才会被调用。也就是说,当 __getattribute__ 查找不到属性(比如抛出 AttributeError),就会尝试调用 __getattr__。
这个方法适合用来实现“动态属性”或默认回退行为。
典型应用场景:
- 为缺失属性提供默认值
- 按需加载某些配置或资源
- 实现类似字典的属性访问(如 obj.undefined 返回 None)
示例:
立即学习“Python免费学习笔记(深入)”;
def __getattr__(self, name):
print(f"属性 {name} 不存在,返回默认值")
return 0
__setattr__(self, name, value)
每当给对象的属性赋值时都会触发,无论是新增还是修改属性。
与 __getattribute__ 类似,如果在该方法中直接使用 self.name = value,会再次触发 __setattr__,导致递归。
推荐写法:
def __setattr__(self, name, value):
print(f"设置属性: {name} = {value}")
super().__setattr__(name, value)
适用场景:
- 限制某些属性不能被修改
- 属性赋值前做类型检查或格式化
- 实现只读属性或受控属性变更通知
基本上就这些。这三个魔法函数配合使用,能让你精细控制对象的属性行为。关键是注意避免无限递归,优先使用 super() 来调用父类实现。











