WeakMap 和 WeakSet 是以弱引用为核心的集合类型,适合处理与对象生命周期绑定的临时数据;WeakMap 键必须为对象,用于存私有状态、DOM 状态或缓存;WeakSet 仅存对象,适用于存在性判断如防递归、事件监听;二者均不支持遍历与 size 等方法,不可替代 Map/Set 作通用集合。

WeakMap 和 WeakSet 是 JavaScript 中两个以“弱引用”为核心的内置集合类型,它们不阻止垃圾回收器释放所持有的对象,因此天然适合处理与对象生命周期强绑定的临时数据。
WeakMap:给对象配专属私有数据
WeakMap 的键必须是对象,且这个引用是弱的——只要该对象在别处不再被引用,它就能被正常回收,WeakMap 里对应的键值对也会自动消失。
- 给 DOM 节点存状态而不影响卸载:比如记录某个按钮是否已初始化,节点被 remove 后,数据自动清理
- 模拟类的私有属性:把实例和它的内部状态映射起来,实例销毁时状态不留痕
- 缓存计算结果:以对象为 key 缓存开销大的处理结果,对象一消失,缓存就失效,不用手动删
WeakSet:快速标记对象状态,不拖后腿
WeakSet 只能存对象,也是弱引用。它不提供遍历能力,但非常适合做“存在性判断”——比如确认某个对象是否已被处理过。
- 防递归遍历:在树或图结构中用 WeakSet 记录已访问节点,对象释放后标记自动清除
- 一次性事件监听:配合 observeOnce 实现“只绑一次”,DOM 元素移除后监听逻辑自然失效
- 活跃对象登记:比如标记当前正在动画中的元素,不需要定时清理过期项
它们共有的关键限制与优势
两者都不支持 size、clear、forEach 或 keys/values/entries 等遍历方法,这是设计使然——正是这种“不可枚举性”保障了弱引用语义不被破坏。
立即学习“Java免费学习笔记(深入)”;
- 不能防止内存泄漏,但能避免制造泄漏
- 无法替代 Map/Set 做通用集合操作,专用于生命周期敏感场景
- 没有键泄漏风险:即使你忘了清理,对象一走,数据就走
什么时候不该用?
当你需要长期持有数据、按需遍历所有条目、或者键可能是字符串/基本类型时,WeakMap 和 WeakSet 就不合适了。这时候老老实实用 Map 或 Set 更稳妥。











