weakMap 和 weakSet 原理

MDN中这样描述:只能将对象、非全局符号作为weakMap的键weakSet的值

两者维持弱引用,即weakMap的键weakSet的值不再引用时,则被内存回收。

应用场景

js 复制代码
const wm = new WeakMap():
const ws = new WeakSet():
let obj = {};
wm.set(obj, 1)
ws.add(obj)
obj = null; 

// 此时 obj 和 obj => 1 都会被内存回收
// (tips:如果是map,obj指向的地址就不会被回收,因为map依旧维持了对其的引用)

weakMap

weakMapkey 被回收时value也被回收,即弱引用

我们需要实现仅 key 可达 value

根据标记清除法,当一个变量不可达时则被回收。

要实现一对一可达,只能使用对象作为key。

如果key是非对象值,则必定需要一个局部对象存储 key => value 键值对,这时 key 不可达 value,而是局部对象可达 value。此时不能实现弱引用,因为 key 被回收,而 value未被回收。

所以, weakMap 可以看做在 key 这个对象类型键上面挂载了 value。这样不管是 weakMap.delete还是 key = nullkey被回收后,value也会被回收。

实现代码:

js 复制代码
class MyWeakMap {
  #key = Symbol();

  set(key, value) {
    key[this.#key] = value;
  }

  get(key) {
    return key[this.#key];
  }

  has(key) {
    return this.#key in key;
  }

  delete(key) {
    delete key[this.#key];
  }
}

weakSet

weakSet同理,想要维持弱引用,就不能使用局部对象去保存value

所以这里的 weakSet 是虚拟的概念,因为想要放在一个set里,必须得有一个局部对象去保存。但是又不能存在局部对象去增加额外的引用,继而破坏弱引用不能回收,所以实际上不存在set集合。

如果想实现虚拟的集合,只能将一组对象组合在一起。因为对象的属性可以保存这组虚拟集合的key,用来判断是否同组。 而非对象值不支持属性,没办法判断分组。所以这也是weakSet为什么只能保存对象、非全局符号。

实现代码:

js 复制代码
class MyWeakSet {
  #key = Symbol();

  add(key) {
    key[this.#key] = true;
  }

  has(key) {
    return !!key[this.#key];
  }

  delete(key) {
    delete key[this.#key];
  }
}
相关推荐
顾林海7 分钟前
Android MMKV 深度解析:原理、实践与源码剖析
android·面试·源码阅读
Java技术小馆7 分钟前
重构 Controller 的 7 个黄金法则
java·后端·面试
timeweaver14 分钟前
深度解析 Nginx 前端 location 配置与优先级:你真的用对了吗?
前端·nginx·前端工程化
鲸落落丶15 分钟前
网络通信---Axios
前端
wwy_frontend17 分钟前
React性能优化实战:从卡顿到丝滑的8个技巧
前端·react.js
小高00732 分钟前
面试官:npm run build 到底干了什么?从 package.json 到 dist 的 7 步拆解
前端·javascript·vue.js
天选打工圣体33 分钟前
个人学习笔记总结(四)抽离elpis并发布npm包
前端
JayceM2 小时前
Vue中v-show与v-if的区别
前端·javascript·vue.js
楼田莉子2 小时前
C++算法题目分享:二叉搜索树相关的习题
数据结构·c++·学习·算法·leetcode·面试
HWL56792 小时前
“preinstall“: “npx only-allow pnpm“
运维·服务器·前端·javascript·vue.js