关于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 function
20       };
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(根)区,只有从这些地方发出的引用,才是可到达的,有效的。

 

posted @ 2015-03-03 11:51  杨元  阅读(674)  评论(2编辑  收藏  举报