
本文深入解析python中类变量在继承体系下的访问与修改机制,阐明为何子类赋值会创建独立属性、而父类赋值会影响未覆盖的子类,帮助开发者避免常见的共享状态陷阱。
在Python中,类变量(如 val = 1)属于类对象本身,被所有实例共享,但在继承链中,其访问和赋值行为遵循“查找优先于继承,赋值即绑定”的核心规则。理解这一点是掌握输出结果的关键。
我们来看原始代码的执行过程:
class A(object):
val = 1
class B(A):
pass
class C(A):
pass
print(A.val, B.val, C.val) # 输出: 1 1 1
B.val = 2
print(A.val, B.val, C.val) # 输出: 1 2 1
A.val = 3
print(A.val, B.val, C.val) # 输出: 3 2 3? 第一行输出 1 1 1:
B 和 C 均未定义自己的 val,因此通过继承链向上查找,最终都访问到 A.val —— 此时三者共享同一份类变量引用。
⚠️ 第二行输出 1 2 1(而非 1 2 2):
B.val = 2 并非修改 A.val,而是在类 B 的命名空间中动态创建(或覆盖)一个名为 val 的新类属性。此后 B.val 指向 B 自身的 val,而 A.val 和 C.val 仍指向 A 的 val。这本质上是属性赋值触发了类字典的写入操作(B.__dict__['val'] = 2),与继承无关。
立即学习“Python免费学习笔记(深入)”;
✅ 第三行输出 3 2 3(而非 3 2 3?等等——它确实是 3 2 3,但原因关键):
A.val = 3 修改的是 A 类自身的 val 属性。由于 C 从未定义 val,它继续通过继承查找 A.val,因此 C.val 现在读取到 3;而 B 已拥有自己的 val(B.__dict__ 中存在 'val': 2),因此查找时不再向上委托,B.val 保持为 2 —— 这正是Python MRO(方法解析顺序)中“就近原则”的体现。
? 关键总结:
- ✅ 读取(attribute lookup):按MRO从左到右搜索,找到即止(C.val → A.val;B.val → B.__dict__ → 命中,不查 A)。
- ❌ 赋值(attribute assignment):总是作用于左侧对象自身的 __dict__(B.val = 2 → 写入 B.__dict__;A.val = 3 → 写入 A.__dict__)。
- ? 类变量不会自动同步;子类显式赋值后即脱离父类变量的“绑定”。
? 实践建议:
若需真正共享可变类变量(如列表、字典),务必谨慎;若意图让所有子类响应父类变更,请避免在子类中直接赋值同名类变量。必要时可通过 @classmethod 封装受控访问逻辑:
class A:
_val = 1
@classmethod
def set_val(cls, v):
cls._val = v # 安全:明确作用于当前类
@classmethod
def get_val(cls):
return cls._val正确理解这一机制,是写出可预测、易维护的面向对象Python代码的重要基础。










