Python优先读取实例属性,因属性查找顺序为实例__dict__先于类__dict__;实例赋值仅修改自身字典,不覆盖类属性;判断来源需分别检查a.__dict__和A.__dict__。
Python 会优先读取实例属性,哪怕同名的类属性存在。这不是“覆盖”,而是属性查找顺序决定的:实例字典 __dict__ 优先于类字典。
关键点在于:只要实例对象的 __dict__ 中有该键,就直接返回值;否则才向上查类及其父类的 __dict__。
常见错误现象:
hasattr(obj, 'x') 返回 True,但 hasattr(cls, 'x') 却是 
False(说明只有实例有)不能单靠打印值来判断,必须检查来源。最直接的方式是分别查看实例和类的 __dict__:
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 = ...,可能无意中让成千上万个实例各自持有一份副本,而不是共享一个类属性。