Python 使用 slots 控制对象内存占用

1次阅读

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

Python 使用 slots 控制对象内存占用

为什么 __slots__ 有时反而让对象变大

不是加了 __slots__ 就一定省内存,尤其当类继承自内置类型(如 listdict)或已有 __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__),整个继承树就废了。

text=ZqhQzanResources