nos*_*tio 8 javascript garbage-collection v8 node.js
我正在V8 中使用WeakRef和FinalizationRegistry,但我无法验证下面的代码是否在 Node.js 中实际工作。
我正在使用 Node v15.3.0 并像这样运行它:
node --expose-gc transient.js:
finalizerCallback called!我希望在控制台日志中看到一些条目。
如果我在基于 Chromium 的浏览器(尝试Run code snippet按钮)中运行它,则在异步脚本仍在运行时,单击 DevTools(在 Chrome v87 中)的性能部分中的垃圾桶图标(收集垃圾),我可以获得该输出。
它在 Firefox v83 中按预期工作,我在脚本结束时看到所有终结器回调。
对于 Node,我希望在显式调用垃圾收集时自动看到它,但终结器回调根本不会被调用。
是代码本身有问题,还是有更可靠的方法在Node中强制GC?
// by @noseratio
// see https://v8.dev/features/weak-references
class Transient {
constructor(eventTarget) {
const finalizerCallback = ({ eventTarget, eventListener }) => {
console.log('finalizerCallback called!');
eventTarget.removeEventListener('testEvent', eventListener);
}
const finalizer = new FinalizationRegistry(finalizerCallback);
const strongRefs = { finalizer };
const weakRef = new WeakRef(this);
const eventListener = () => {
console.log('eventListener called!');
strongRefs.finalizer = null;
weakRef.deref()?.test();
}
finalizer.register(this, { eventTarget, eventListener });
eventTarget.addEventListener('testEvent', eventListener, { once: true });
}
test() {
console.log("test called!");
}
}
async function main() {
const gc = globalThis?.global?.gc;
console.log(`garbage collector func: ${gc}`);
const eventTarget = new EventTarget();
for (let i = 10; i > 0; i--) {
void function () {
// these instances of Transient really must be getting GC'ed!
new Transient(eventTarget);
}();
await new Promise(r => setTimeout(() => r(gc?.(true)), 100));
}
console.log("finishing...")
gc?.(true);
await new Promise(r => setTimeout(r, 5000));
eventTarget.dispatchEvent(new Event("testEvent"));
console.log("done.")
}
main().catch(console.error);Run Code Online (Sandbox Code Playgroud)
更新后,如果我增加循环集成的数量for,我最终会看到一些finalizerCallback调用,但它们仍然是零星的。
我直接从@jasnell得到了答案:
您可以尝试本机语法功能
%CollectGarbage
我已经尝试过,它完全按照我想要的方式工作。在现实生活中,它是一个对象链AbortController,这里有更多上下文。
启用%CollectGarbage:
node --allow-natives-syntax Transient.js
Run Code Online (Sandbox Code Playgroud)
%要对静态代码分析器隐藏语法,我们可以使用eval:
eval("%CollectGarbage('all')");
Run Code Online (Sandbox Code Playgroud)
| 归档时间: |
|
| 查看次数: |
1306 次 |
| 最近记录: |