__getattr__用于处理访问不存在的属性时的逻辑,如动态计算、代理或延迟加载;它在属性查找失败后被调用,适合复杂场景,而非常规取值。

Python 获取对象属性,通常情况下直接用
.操作符就能搞定。但如果属性名是动态的,或者你想在访问属性时做点额外的事情,
__getattr__函数就派上用场了。它就像一道后备防线,当 Python 找不到你请求的属性时,就会调用它。
解决方案:
__getattr__是 Python 类中的一个特殊方法(也称为魔术方法)。当试图访问一个对象不存在的属性时,Python 解释器会调用这个方法。你需要做的就是在你的类中定义
__getattr__方法,并编写处理未找到属性的代码。
class MyClass:
def __init__(self, data):
self._data = data
def __getattr__(self, name):
if name.startswith('computed_'):
# 假设 computed_ 开头的属性需要计算
key = name[len('computed_'):] # 提取实际的 key
if key in self._data:
return self._data[key] * 2 # 简单计算示例
else:
raise AttributeError(f"属性 {name} 不存在")
else:
raise AttributeError(f"属性 {name} 不存在")
# 示例用法
data = {'x': 10, 'y': 20}
obj = MyClass(data)
print(obj.computed_x) # 输出: 20
print(obj.computed_y) # 输出: 40
# 访问不存在的属性
try:
print(obj.computed_z)
except AttributeError as e:
print(e) # 输出: 属性 computed_z 不存在
try:
print(obj.normal_attribute)
except AttributeError as e:
print(e) # 输出: 属性 normal_attribute 不存在
__getattr__接收一个参数
name,它就是你试图访问但不存在的属性名。在
__getattr__方法中,你可以根据
name来决定如何处理。上面的例子展示了如何根据属性名前缀来动态计算属性值,如果属性不存在,则抛出
AttributeError异常,这是良好的实践。
立即学习“Python免费学习笔记(深入)”;
如果仅仅是简单地想从一个字典里取值,并且字典里没有对应的key时,返回一个默认值,其实有更简洁的方式,比如使用
getattr的第三个参数,或者直接使用字典的
get方法。
__getattr__更适合处理更复杂的逻辑,比如动态计算属性,或者根据某种规则返回不同的值。
__getattr__和
__getattribute__有什么区别?
__getattribute__会拦截所有属性的访问,包括存在的属性。这意味着如果你定义了
__getattribute__,那么每次访问属性时都会先调用它,然后再决定是否返回属性值或调用
__getattr__。因此,在使用
__getattribute__时要格外小心,避免无限递归。通常情况下,
__getattr__更安全、更常用。
什么时候应该使用
__getattr__?
- 动态属性: 当你需要根据某些规则动态地创建属性时。
- 属性代理: 当你希望将属性访问委托给其他对象时。
- 延迟加载: 当你希望在真正访问属性时才加载其值时。
如何避免
__getattr__引起的性能问题?
__getattr__会在每次属性查找失败时被调用,这可能会影响性能。为了避免性能问题,可以考虑以下几点:
- 缓存结果: 如果属性的计算成本很高,可以将结果缓存起来,下次直接返回缓存值。
- 减少调用次数: 尽量避免频繁访问不存在的属性。
-
使用 slots: 使用
__slots__
可以防止动态创建属性,从而避免调用__getattr__
。
总的来说,
__getattr__是一个强大的工具,可以让你灵活地处理属性访问。但是,要谨慎使用,避免滥用导致代码难以理解和维护。










