JavaScript WeakMap 不断引用经过 gc 处理的对象

2023-11-25

我正在经历 JavaScript 弱映射,在 google chrome 开发者控制台中尝试此代码后,使用 --js-flags="--expose-gc" 运行,我不明白为什么弱映射继续引用 a.b 如果 a 是GC 编辑。

var a = {listener: function(){ console.log('A') }}
a.b = {listener: function(){ console.log('B') }}

var map = new WeakMap()

map.set(a.b, [])
map.set(a, [a.b.listener])

console.log(map) // has both a and a.b

gc()
console.log(map) // still have both a and a.b

a = undefined
gc()
console.log(map) // only have a.b: why does still have a reference to a.b? Should'nt be erased?

更新 2/2020

当我现在运行这段代码时,它按预期工作。我认为在以前版本的 Chrome 中打开控制台会导致对象被保留,但现在不会。重新分配保存对象引用的变量的值将导致该对象被垃圾收集(假设没有其他对象引用它)。


在您的示例代码中,您没有释放您的a多变的。它是一个顶级变量,永远不会超出范围,也永远不会被显式取消引用,因此它保留在 WeakMap 中。一旦代码中不再有对象的引用,WeakMap/WeakSet 就会释放该对象。在你的例子中,如果你console.log(a)在你的一个之后gc()打电话,你仍然会期待a活着,对吗?

下面是一个工作示例,展示了 WeakSet 的实际作用,以及一旦对某个条目的所有引用都消失了,它将如何删除该条目:https://embed.plnkr.co/cDqi5lFDEbvmjl5S19Wr/

const wset = new WeakSet();

// top level static var, should show up in `console.log(wset)` after a run
let arr = [1];
wset.add(arr);

function test() {
  let obj = {a:1}; //stack var, should get GCed
  wset.add(obj);
}

test();

//if we wanted to get rid of `arr` in `wset`, we could explicitly de-reference it
//arr = null;

// when run with devtools console open, `wset` always holds onto `obj`
// when devtools are closed and then opened after, `wset` has the `arr` entry,
// but not the `obj` entry, as expected
console.log(wset);

请注意,打开 Chrome 开发工具会阻止某些对象被垃圾收集,这使得看到这一点比预期更困难:)

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

JavaScript WeakMap 不断引用经过 gc 处理的对象 的相关文章