与往常一样,MDN 的文档帮助.
WeakRef 对象包含对对象的弱引用,该对象称为其目标或引用对象。对对象的弱引用是不会阻止对象被垃圾收集器回收的引用。相反,普通(或强)引用将对象保存在内存中。当一个对象不再有任何强引用时,JavaScript 引擎的垃圾收集器可能会销毁该对象并回收其内存。如果发生这种情况,您将无法再从弱引用中获取对象。
在 JS 的几乎所有其他部分中,如果某个对象 (A) 持有对另一个对象 (B) 的引用,则 B 不会被垃圾回收,直到 A 也可以被完全垃圾回收。例如:
// top level
const theA = {};
(() => {
// private scope
const theB = { foo: 'foo' };
theA.obj = obj;
})();
在这种情况下,theB
永远不会被垃圾回收(除非theA.obj
被重新分配),因为theA
在顶层包含一个持有对theB
;的引用的属性。这是一个强引用,可以防止垃圾收集。
另一方面,WeakRef 提供了一个可以访问对象的包装器,同时不会阻止对该对象进行垃圾回收。如果对象尚未被垃圾回收,则调用deref()
WeakRef 将返回给您。如果它已被 GC 处理,将返回..deref()
undefined
FinalizationRegistry处理类似的问题:
FinalizationRegistry 对象允许您在对象被垃圾回收时请求回调。
您首先使用要运行的回调定义注册表,然后.register
使用要观察的对象调用注册表。这将让您确切地知道何时收集垃圾。例如,Just got GCd!
一旦obj
被回收,将记录以下内容:
console.log('script starting...');
const r = new FinalizationRegistry(() => {
console.log('Just got GCd!');
});
(() => {
// private closure
const obj = {};
r.register(obj);
})();
您还可以在调用时传递一个值,该值在.register
收集对象时传递给回调。
new FinalizationRegistry((val) => {
console.log(val);
});
r.register(obj, 'the object named "obj"')
将记录the object named "obj"
它得到GC'd。
综上所述,很少需要这些工具。正如 MDN 所说:
正确使用 FinalizationRegistry 需要仔细考虑,如果可能,最好避免使用。避免依赖规范未保证的任何特定行为也很重要。何时、如何以及是否发生垃圾收集取决于任何给定 JavaScript 引擎的实现。您在一个引擎中观察到的任何行为可能在另一个引擎中、在同一引擎的另一个版本中、甚至在同一引擎的相同版本中略有不同的情况下有所不同。垃圾收集是 JavaScript 引擎实现者不断完善和改进其解决方案的难题。
最好尽可能让引擎本身自动处理垃圾收集,除非您有充分的理由自己关心它。