jQuery.data与原始 expando 属性(可以分配给 DOM 节点的任意属性)相比,被吹捧的优势之一是jQuery.data “不受循环引用的影响,因此不会出现内存泄漏”。Google 的一篇题为“优化 JavaScript 代码”的文章更详细地介绍了:
Web 应用程序最常见的内存泄漏涉及 JavaScript 脚本引擎和实现 DOM 的浏览器的 C++ 对象之间的循环引用(例如,在 JavaScript 脚本引擎和 Internet Explorer 的 COM 基础结构之间,或 JavaScript 引擎和 Firefox XPCOM 基础结构之间)。
它列出了循环引用模式的两个示例:
DOM 元素 → 事件处理程序 → 闭包范围 → DOM
DOM 元素 → 通过 expando → 中间对象 → DOM 元素
但是,如果 DOM 节点和 JavaScript 对象之间的引用循环产生了内存泄漏,这是否意味着任何重要的事件处理程序(例如onclick
)都会产生这种泄漏?我看不出事件处理程序如何避免引用循环,因为我看到它的方式是:
DOM 元素引用事件处理程序。
事件处理程序引用 DOM(直接或间接)。无论如何,要避免
window
在任何有趣的事件处理程序中引用几乎是不可能的,除非编写一个setInterval
从全局队列中读取动作的循环。
有人可以提供 JavaScript ↔ DOM 循环引用问题的准确解释吗?我想澄清的事情:
哪些浏览器受到影响?jQuery 源代码中的一条评论特别提到了 IE6-7,但 Google 文章暗示 Firefox 也受到了影响。
expando 属性和事件处理程序在内存泄漏方面是否有所不同?或者这两个代码片段都容易受到同一种内存泄漏的影响?
// Create an expando that references to its own element. var elem = document.getElementById('foo'); elem.myself = elem; // Create an event handler that references its own element. var elem = document.getElementById('foo'); elem.onclick = function() { elem.style.display = 'none'; };
如果页面由于循环引用而泄漏内存,泄漏是否会持续到整个浏览器应用程序关闭,或者当窗口/选项卡关闭时内存是否被释放?