首页 > 网站 > WEB开发 > 正文

关于JavaScript内存泄漏的质疑

2024-04-27 14:12:27
字体:
来源:转载
供稿:网友

关于javaScript内存泄漏的质疑

近几天看了些关于Javascript内存管理的文章,相对于Java JVM的内存管理,显得简单些。

在学习的过程中,发现有不少网友谈到了循环引用,说循环引用会造成内存泄漏,垃圾回收器无法回收。

实际上,并没有这么可怕,根据小菜目前的了解,这种循环引用造成的内存泄漏,仅仅会发生在低版本的IE浏览器上,现代浏览器是不会这么蠢的。

举个例子,网络上流行的说法大致有如下两种:

 1 <!DOCTYPE html> 2 <html> 3   <head> 4     <meta charset="utf-8"> 5     <meta name="viewport" content="width=640, initial-scale=0.5, user-scalable=no" /> 6     <title>循环引用内存分析</title> 7     <style> 8     </style> 9   </head>10   <body>11     <input type="button" onclick="PRoblem();" value="call Problem">12     <input type="button" onclick="MyBindEvent();" value="call MyBindEvent">13   </body>14   <script>15     //闭包引起的隐式循环引用16     function MyBindEvent(){17       var obj=document.createElement("div");18       obj.onclick=function(){19         //Even if it's a empty function20       };21     }22 23     //显式循环引用24     function Problem() {25       var objA = new Object();26       var objB = new Object();27 28       objA.someOtherObject = objB;29       objB.anotherObject = objA;30     }31   </script>32 33 </html>

一个简单的页面,上边两个按钮,分别调用两个会造成内存泄漏的方法。

借助于Chrome浏览器的Profiles功能,生成内存快照,然后对比,发现这两种写法在谷歌浏览器下均没有泄漏问题。

具体做法是:

  1. 打开页面不做任何操作,直接生成页面内存快照。
  2. 点击按钮,然后再次生成内存快照。
  3. 对比两次内存变化。

不断重复这个过程,生成7、8个快照,趋于稳定,会发现往后内存根本没有变化。

每次生成快照之前,都会强制执行GC(垃圾回收),说明我们每次构造的循环引用,马上被回收了,所以不会出现在快照中。

接下来从理论角度说说为什么应该被回收。

因为这些循环引用,说白了都是无效引用。可以简单理解为:只有从栈区发起的引用才是有效的。本例中的引用,是堆区对象的互相引用,虽然引用计数不为0,但是不可到达,在回收内存时直接就被消灭了。

再深入了说,低版本IE浏览器采用的是引用计数机制回收内存,互相引用造成对方计数互不为0,导致无法回收。

而现代浏览器,采用的是Cheney算法,大致就是把内存分为两份,不断的来回复制,这样那些不可到达的对象,就无法复制,自然被回收了。

栈区、静态、常量之类的字眼,一般是代表root(根)区,只有从这些地方发出的引用,才是可到达的,有效的。


发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表