__slots__ 并非总省内存:继承内置类型或含__dict__的父类时会被忽略,反而可能增大实例;仅对纯新式类且无父类__dict__时生效,需用hasattr(parent, ‘__dict__’)检查。

为什么 __slots__ 有时反而让对象变大
不是加了 __slots__ 就一定省内存,尤其当类继承自内置类型(如 list、dict)或已有 __dict__ 的父类时,python 会直接忽略 __slots__。这种情况下不仅不省空间,还可能因额外的元信息导致实例更大。
- 只对「纯新式类」且「无父类定义
__dict__」时生效;检查父类是否含__dict__:用hasattr(Parent, '__dict__') -
__slots__声明的字段名必须是字符串序列,写成__slots__ = 'x', 'y'(缺括号)会变成单个字符串,导致只允许一个属性'x',其余赋值抛AttributeError - 如果类需要动态添加属性(比如打日志、临时缓存),硬上
__slots__会导致运行时报错,得提前预留'__dict__'—— 但这就基本抵消内存收益
如何验证 __slots__ 真正起效
别光看代码写了没,得测实际内存和结构。最直接的方式是对比 sys.getsizeof() 和 obj.__dict__ 是否存在。
- 加了
__slots__后,实例不应有__dict__:hasattr(inst, '__dict__')应为False - 用
vars(inst)查看属性容器:返回TypeError表示__slots__生效;若返回字典,说明没生效或被绕过 - 注意
sys.getsizeof()只算对象本身,不含引用内容(如字符串、列表),要测整体开销得结合pympler.asizeof
__slots__ 和 Property / 描述符共存要注意什么
__slots__ 只控制实例属性存储位置,不影响描述符逻辑,但容易误以为「property 名也要放进 __slots__」。
-
@property的 getter/setter 是类属性,不占实例空间,无需、也不该写进__slots__ - 如果 property 背后依赖一个私有实例变量(如
_value),那_value才是需声明在__slots__中的项 - 使用
__slots__后,无法通过inst.__dict__['x'] = val绕过限制,所有赋值都走__setattr__,所以自定义__setattr__时得小心处理 slots 字段校验
多继承下 __slots__ 的冲突表现
多个父类都定义了 __slots__,但内容不一致时,子类会报 TypeError: multiple bases have instance lay-out conflict。
立即学习“Python免费学习笔记(深入)”;
- 只要任一父类有
__slots__,其他同级父类也必须有,且不能有重叠字段(除非完全一致) - 常见踩坑:混用第三方库类(如
dataclasses生成的类默认无__slots__)和你自己加了__slots__的基类 - 解决办法只有两个:要么统一所有父类加相同
__slots__,要么放弃继承,改用组合(composition)
实际用的时候,最容易被忽略的是「父类有没有悄悄带 __dict__」——哪怕它没显式定义,只要继承链里某一级用了普通类定义(没设 __slots__),整个继承树就废了。