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 正确回收。

相关推荐
码事漫谈18 小时前
时序数据库2026盘点:国产数据库如何以“融合多模”走出差异化之路?
前端·后端
道友可好18 小时前
让 AI 自己验收,等于让学生自己批卷
前端·人工智能·后端
yingyima18 小时前
Go 语言正则表达式速查手册:30 分钟掌握核心语法与实战技巧
前端
大蝴蝶博努奇a18 小时前
使用ChatGPT 解决各类代码报错
前端
胡志辉18 小时前
深入浅出 call、apply、bind
前端·javascript·后端
iccb101319 小时前
5年,一个程序员是如何把私有化在线客服系统做到第一名的
前端·后端·github
假如让我当三天老蒯19 小时前
回归基本功:Map/Set 与 WeakMap/WeakSet 的区别
前端·面试
IT乐手19 小时前
48队都装不下你|国足第24次让全世界失望
前端
SoaringHeart20 小时前
Flutter最佳实践:IM聊天文字链接自动识别跳转
前端·flutter
掘金一周21 小时前
企业中要做智能体,最佳的方案是什么? | 沸点周刊 6.18
前端·人工智能·ai编程