8

我需要一些帮助来了解它是如何工作的(或者没有,就此而言)。

  1. 在网页中,我为节点创建了一个单击事件侦听器。
  2. 在侦听器中,我创建了某个随机类的实例,它将节点设置为自身内部的属性。因此,如果var classInstance是实例,我可以访问节点,例如classInstance.rootNode.
  3. 当监听器触发时,我设置了一个 ajax 请求,保持classInstance关闭并将 ajax 响应传递给classInstance并使用它来修改rootNode' 的样式或内容或其他任何东西。

我的问题是,一旦我完成了classInstance,假设没有其他引用它并且它本身在它自己的闭包中没有任何其他内容,垃圾收集器会处理它吗?如果没有,我如何标记它以进行处置?

4

2 回答 2

7

为了回应@Beetroot-Beetroot 的疑虑(诚然,我也有),我做了更多的挖掘工作。我设置了这个小提琴,并使用了chrome 开发工具的时间线这篇文章作为指导。在小提琴中,两个几乎相同的处理程序创建了一个带有 2 个日期对象的闭包。第一个仅引用a,第二个引用都ab。尽管在这两种情况下都只能a真正暴露(硬编码值),但第一个闭包使用的内存要少得多。这是否取决于 JIC(即时编译)或 V8 的 JS 优化魔法,我不能肯定地说。但从我读过的内容来看,我会说是 V8 的 GCbtst函数返回,在第二种情况下它不能返回(返回时bar引用)。我觉得这并不是那么古怪,而且我一点也不惊讶发现 FF 甚至 IE 也能以类似的方式工作。刚刚添加了这个,也许是无关紧要的,为了完整性而更新,因为我觉得谷歌开发工具文档的链接似乎是一种附加值btst2


它有点取决于,作为一个简单的经验法则:只要你不能再引用classInstance变量,它就应该被 GC'ed,不管它自己的循环引用。我已经测试了很多结构,类似于您在此处描述的结构。也许值得一看,
我发现闭包和内存泄漏并不那么常见或容易得到(至少,现在不是了)。

但正如公认的答案所说:几乎不可能知道什么时候会泄漏什么代码。
再次阅读您的问题,我会说:不,您不会泄漏内存:该classInstance变量不是在全局范围内创建的,而是被传递给各种函数(因此是各种范围)。每次函数返回时,这些作用域都会分解。classInstance如果已将其传递给另一个函数/范围,则不会被 GC。但是一旦引用的最后一个函数classInstance返回,该对象就会被标记为 GC。当然它可能是一个循环引用,但它是一个除了它自己的范围之外不能从任何地方访问的引用。
您也不能真正将其称为闭包:当存在某种形式的暴露时,就会发生闭包到外部范围,这在您的示例中没有发生。

我在解释这样的事情时很垃圾,但只是回顾一下:

var foo = (function()
{
    var a, b, c, d;
    return function()
    {
        return a;
    }
})();

GC 将释放 memb和引用:它们已经超出范围,无法访问它们c......d

var foo = (function()
{
    var a, b, c, d;
    return function()
    {
        a.getB = function()
        {
            return b;
        }
        a.getSelf = function()
        {
            return a;//or return this;
        }
        return a;
    }
})();
//some code
foo = new Date();//

在这种情况下,b由于显而易见的原因,也不会被 GC 处理。foo暴露ab,其中a是一个包含循环引用的对象。虽然一出,就foo = new Date()失去foo了任何参考a。当然,a仍然引用自己,但a不再暴露:它可以引用任何它非常喜欢的东西。大多数浏览器不会在意并且会 GCab. 事实上,我已经检查过 Chrome、FFIE8 都完美地 GC 上面的代码......那么不用担心。

于 2012-09-27T11:15:07.973 回答
2

我不是这个问题的专家,但我很确定 GC不会处理它。事实上,它可能永远不会,因为您已经在事件侦听器和 DOM 节点之间创建了一个循环引用。要允许它被垃圾收集,您应该将这些引用(事件侦听器和/或 rootNode)中的一个或两个设置为 undefined 或 null。

不过,如果您要创建许多这样的类实例,或者如果它可以在页面的生命周期内创建多次,我只会担心这一点。否则这是一个不必要的优化。

于 2012-09-27T10:55:52.170 回答