Python优先读取实例属性,因属性查找顺序为实例__dict__先于类__dict__;实例赋值仅修改自身字典,不覆盖类属性;判断来源需分别检查a.__dict__和A.__dict__。

类属性和实例属性同名时,读取优先级是什么
Python 会优先读取实例属性,哪怕同名的类属性存在。这不是“覆盖”,而是属性查找顺序决定的:实例字典 __dict__ 优先于类字典。
关键点在于:只要实例对象的 __dict__ 中有该键,就直接返回值;否则才向上查类及其父类的 __dict__。
常见错误现象:
- 误以为修改了类属性,实际只改了某个实例的同名属性
- 动态给实例赋值后,再访问该名字,发现类属性“失效”了
- 用
hasattr(obj, 'x')返回True,但hasattr(cls, 'x')却是False(说明只有实例有)
如何判断当前读到的是类属性还是实例属性
不能单靠打印值来判断,必须检查来源。最直接的方式是分别查看实例和类的 __dict__:
立即学习“Python免费学习笔记(深入)”;
class A:
x = 10
a = A()
print('x in a.dict?', 'x' in a.dict) # False
print('x in A.dict?', 'x' in A.dict) # True
a.x = 20
print('x in a.dict?', 'x' in a.dict) # True
print('a.x =', a.x) # 20(来自实例)
print('A.x =', A.x) # 10(类属性未变)
注意:getattr(a, 'x') 永远返回最终查找到的值,不告诉你来源;而 a.__dict__.get('x') 只查实例层,A.__dict__.get('x') 只查类层。
修改同名属性时,到底改了谁
取决于你操作的对象和方式:
-
a.x = 99→ 写入实例__dict__,不影响类属性 -
A.x = 99→ 修改类属性,所有未设置实例属性的实例都会看到新值 -
del a.x→ 删除实例属性,之后再读a.x就会回退到类属性 -
del A.x→ 删除类属性,若实例没定义x,再读会抛AttributeError
性能影响:实例属性查找比类属性快,因为少一层字典遍历;但滥用同名覆盖会让逻辑变隐晦,尤其在继承链中容易误判数据归属。
什么时候该避免同名,什么时候可以接受
同名本身合法,但是否合理要看语义:
- 适合同名:类属性作默认值(如
default_timeout = 30),实例按需覆盖(self.timeout = 5) - 应避免同名:类属性用于共享状态(如计数器
count = 0),又在实例方法里写self.count += 1—— 这会悄悄创建实例属性,导致计数失效 - 调试建议:对关键共享属性,用
@classmethod或显式通过cls.访问,避免self.模糊语义
最容易被忽略的是:实例赋值触发属性创建这一行为不可逆(除非手动 del),且不会警告。一旦你在循环中反复执行 obj.attr = ...,可能无意中让成千上万个实例各自持有一份副本,而不是共享一个类属性。










