ES6 WeakMap 生效的证明: FinalizationRegistry

为什么要有 Map 和 WeakMap?

普通对象的 key只能是 字符串 或者 Symbol

没有办法使用 Object 作为对象。

和传统的对象比,Map和WeakMap的优点

  • map中的各映射是有顺序的,object是没有
  • 传统的object对象只能用【字符串、Symbol】作key(键值),而map甚至可以用对象作key值。
  • 上面的第二点其实会造成很严重的问题,这意味着我们没法区分 1 和 '1'

WeakMap - JavaScript | MDN

【弱引用】:在没有其他引用的时候,垃圾回收正常进行。

WeakMap 垃圾回收演示

javascript 复制代码
const register = new FinalizationRegistry((value) => {
  console.log('🍀🍀🍀🍀 回收了', value)
})

let key = {
  obj: 'chp',
}

const m = new WeakMap()

m.set(key, 1)

register.register(key)

key = null

gc() // 🍀🍀🍀🍀 回收了 undefined

对于 WeakMap 来说,__obj__由于通过标记清除的方式,从全局对象找不到了,WeakMap的引用不会影响循环引用对象的【被清理】。

有利于节约内存。

并且本身,WeakMap的占用就比 Map 要小。(毕竟是阉割版本的 Map,只有 get/set/has/delete)。

如官网所说,就像上面的例子一样。

如果 obj 本身就是你想要清除的。

然后又默认触发了GC(假设自动发生了这件事情)。

那么你通过 WeakMap.get 就得不到你想要的值了 => 这种时候你要用 Map 。

应用场景

  • 深拷贝------循环引用
  • 性能需求较高的场景

WeakMap 更节省空间的证明

占用的变量更小

scss 复制代码
function printMermory() {
  let used = 0

  return function (isPrint) {
    let nowUsed = process.memoryUsage().heapUsed
    if (isPrint) {
      console.log('🍀🍀🍀🍀 总共减少了', (used - nowUsed) / 1024 / 1024)
    } else {
      used = nowUsed
    }
  }
}

const cost = printMermory()

function testMap() {
  const map = new Map()
  let keys = []
  for (let i = 0; i < 10000; i++) {
    const key = { index: i }
    keys.push(key)
    map.set(key, 1)
  }
  cost()
  keys = null
  setTimeout(() => {
    gc()
  }, 0)

  setTimeout(() => {
    console.log('🍀🍀🍀🍀', '内存释放之后')
    cost(true)
  }, 3000)
}

function testWeakMap() {
  const map = new WeakMap()
  let keys = []
  for (let i = 0; i < 10000; i++) {
    const key = { index: i }
    keys.push(key)
    map.set(key, 1)
  }
  cost()
  keys = null
  setTimeout(() => {
    gc()
  }, 0)
  setTimeout(() => {
    console.log('🍀🍀🍀🍀', '内存释放之后')
    cost(true)
  }, 3000)
}

// testMap() // 总共减少了 1.170097351074218

testWeakMap() // 总共减少了 0.75359

分别在 Node 环境中执行上面的代码,可以看到,WeakMap 占据的空间更少

css 复制代码
node --expose-gc weakMap3.js

弱引用的证明

javascript 复制代码
const register = new FinalizationRegistry((value) => {
  console.log('🍀🍀🍀🍀 回收了', value)
})

let key = {
  obj: 'chp',
}
register.register(key)

const weakIns = new WeakMap()
// weakIns.set(key, 1) // 当然,这里的 key 是可以被回收的

const ins = new Map()
// ins.set(key, 1) // 可以看到,只有 weakMap 会被回收

key = null

gc() // 🍀🍀🍀🍀 回收了 undefined

我们轮流打开 weakIns 和 ins。【注意不要同时打开,因为只要 Map() 打开了,key 为 null 的时候,key是不会被回收的。因为由于引用计数的原因,此时 key 被 Map 引用】

而 WeakMap 就不会有这个问题。

注释 Map,可以发现 当 key = null 的时候,直接被 gc 正确回收。

相关推荐
_骁1 分钟前
记两次谷歌浏览器升级引起的bug
前端
轻语呢喃11 分钟前
DeepSeek 接口调用:从 HTTP 请求到智能交互
javascript·deepseek
风之舞_yjf42 分钟前
Vue基础(14)_列表过滤、列表排序
前端·javascript·vue.js
belldeep1 小时前
QuickJS 如何发送一封邮件 ?
javascript·curl·smtp·quickjs
BillKu1 小时前
scss(sass)中 & 的使用说明
前端·sass·scss
疯狂的沙粒1 小时前
uni-app 项目支持 vue 3.0 详解及版本升级方案?
前端·vue.js·uni-app
Jiaberrr2 小时前
uniapp Vue2 获取电量的独家方法:绕过官方插件限制
前端·javascript·uni-app·plus·电量
谢尔登2 小时前
【React】React 18 并发特性
前端·react.js·前端框架
Joker`s smile2 小时前
使用React+ant Table 实现 表格无限循环滚动播放
前端·javascript·react.js
国家不保护废物2 小时前
🌟 React 魔法学院入学指南:从零构建你的第一个魔法阵(项目)!
前端·react.js·架构