Python中类变量被所有实例共享,修改可变类变量(如列表、字典)会意外影响其他实例;应将可变数据移至__init__中用self初始化为实例变量,不可变类变量(如字符串、数字)适合存常量或配置。

Python中类变量(类属性)被所有实例共享,修改可变类属性会影响所有实例,这是常见陷阱。
类变量和实例变量的区别
类变量定义在类内部、方法外部,属于类本身;实例变量在__init__中用self绑定,每个实例独有一份。
关键区别在于:不可变类型(如数字、字符串)的类变量被“覆盖”时实际创建了同名实例变量,不影响其他实例;而可变类型(如列表、字典)被原地修改(如append()、update())时,所有实例看到的是同一对象。
典型出错场景:可变类变量被意外修改
以下代码会出问题:
立即学习“Python免费学习笔记(深入)”;
class Counter:
items = [] # ❌ 可变类变量,所有实例共享
<p>c1 = Counter()
c2 = Counter()
c1.items.append("a")
print(c2.items) # 输出 ['a'] —— 意外!
</p>- 应改用实例变量:self.items = [] 在 __init__ 中初始化
- 若真需类级计数器等不可变值,可用 int、str 等类型,但注意赋值操作(+= 或 =)会生成新对象,不共享
如何安全地使用类变量
类变量适合存放常量、配置项或真正需要全局共享的不可变数据:
- ✅ VERSION = "1.2"、MAX_RETRY = 3
- ✅ 用类方法管理共享状态(如缓存),但内部确保线程安全或明确文档说明
- ❌ 避免直接将 list、dict、set 等作为类变量用于实例间隔离场景
快速检测和修复技巧
检查类中是否有可变对象字面量([]、{}、set())直接赋值给类属性。
修复方式统一为:移到 __init__ 中初始化,或用 None 占位 + 延迟初始化:
class SafeCounter:
items = None # ✅ 类变量仅作标记
<pre class="brush:php;toolbar:false;">def __init__(self):
if self.items is None:
self.items = [] # ✅ 每次实例化新建
更推荐直接写 self.items = [],简洁明确。









