常见的 JavaScript 内存泄露

什么是内存泄露

内存泄漏指由于疏忽或错误造成程序未能释放已经不再使用的内存。内存泄漏并非指内存在物理上的消失,而是应用程序分配某段内存后,由于设计错误,导致在释放该段内存之前就失去了对该段内存的控制,从而造成了内存的浪费。内存泄漏通常情况下只能由获得程序源代码的程序员才能分析出来。然而,有不少人习惯于把任何不需要的内存使用的增加描述为内存泄漏,即使严格意义上来说这是不准确的。————wikipedia

⚠️注:下文中标注的CG是Chrome浏览器中Devtools的【Collect garbage】按钮缩写,表示回收垃圾操作。

意外的全局变量

JavaScript对未声明变量的处理方式:在全局对象上创建该变量的引用(即全局对象上的属性,不是变量,因为它能通过 delete删除)。如果在浏览器中,全局对象就是window对象。

如果未声明的变量缓存大量的数据,会导致这些数据只有在窗口关闭或重新刷新页面时才能被释放。这样会造成意外的内存泄漏。

  
  
  1. function foo(arg) {

  2.    bar = "this is a hidden global variable with a large of data";

  3. }

等同于:

  
  
  1. function foo(arg) {

  2.    window.bar = "this is an explicit global variable with a large of data";

  3. }

另外,通过this创建意外的全局变量:

  
  
  1. function foo() {

  2.    this.variable = "potential accidental global";

  3. }

  4. // 当在全局作用域中调用foo函数,此时this指向的是全局对象(window),而不是'undefined'

  5. foo();

解决方法:

在JavaScript文件中添加 'use strict',开启严格模式,可以有效地避免上述问题。

  
  
  1. function foo(arg) {

  2.    "use strict" // 在foo函数作用域内开启严格模式

  3.    bar = "this is an explicit global variable with a large of data";// 报错:因为bar还没有被声明

  4. }

如果需要在一个函数中使用全局变量,可以像如下代码所示,在window上明确声明:

  
  
  1. function foo(arg) {

  2.    window.bar = "this is a explicit global variable with a large of data";

  3. }

这样不仅可读性高,而且后期维护也方便

谈到全局变量,需要注意那些用来临时存储大量数据的全局变量,确保在处理完这些数据后将其设置为null或重新赋值。全局变量也常用来做cache,一般cache都是为了性能优化才用到的,为了性能,最好对cache的大小做个上限限制。因为cache是不能被回收的,越高cache会导致越高的内存消耗。

console.log

console.log:向web开发控制台打印一条消息,常用来在开发时调试分析。有时在开发时,需要打印一些对象信息,但发布时却忘记去掉 console.log语句,这可能造成内存泄露。

在传递给 console.log的对象是不能被垃圾回收 ♻️,因为在代码运行之后需要在开发工具能查看对象信息。所以最好不要在生产环境中 console.log任何对象。

实例------>demos/log.html
  
  
  1. <!DOCTYPE html>

  2. <html lang="en">

  3. <head>

  4.  <meta charset="UTF-8">

  5.  <meta name="viewport" content="width=device-width, initial-scale=1.0">

  6.  <meta http-equiv="X-UA-Compatible" content="ie=edge">

  7.  <title>Leaker</title>

  8. </head>

  9. <body>

  10.  <input type="button" value="click">

  11.  <script>

  12.    !function () {

  13.      function Leaker() {

  14.        this.init();

  15.      };

  16.      Leaker.prototype = {

  17.        init: function () {

  18.          this.name = (Array(100000)).join('*');

  19.          console.log("Leaking an object %o: %o", (new Date()), this);// this对象不能被回收

  20.        },

  21.        destroy: function () {

  22.          // do something....

  23.        }

  24.      };

  25.      document.querySelector('input').addEventListener('click', function () {

  26.        new Leaker();

  27.      }, false);

  28.    }()

  29.  </script>

  30. </body>

  31. </html>

这里结合Chrome的Devtools–>Performance做一些分析,操作步骤如下:

⚠️注:最好在隐藏窗口中进行分析工作,避免浏览器插件影响分析结果

  1. 开启【Performance】项的记录

  2. 执行一次CG,创建基准参考线

  3. 连续单击【click】按钮三次,新建三个Leaker对象

  4. 执行一次CG

  5. 停止记录

可以看出【JS Heap】线最后没有降回到基准参考线的位置,显然存在没有被回收的内存。如果将代码修改为:

  
  
  1.    !function () {

  2.      function Leaker() {

  3.        this.init();

  4.      };

  5.      Leaker.prototype = {

  6.        init: function () {

  7.          this.name = (Array(100000)).join('*');

  8.        },

  9.        destroy: function () {

  10.          // do something....

  11.        }

  12.      };

  13.      document.querySelector('input').addEventListener('click', function () {

  14.        new Leaker();

  15.      }, false);

  16.    }()

去掉 console.log("Leaking an object %o: %o",(newDate()),this);语句。重复上述的操作步骤,分析结果如下:

从对比分析结果可知, console.log打印的对象是不会被垃圾回收器回收的。因此最好不要在页面中 console.log任何大对象,这样可能会影响页面的整体性能,特别在生产环境中。除了 console.log外,另外还有 console.dirconsole.errorconsole.warn等都存在类似的问题,这些细节需要特别的关注。

closures(闭包)

当一个函数A返回一个内联函数B,即使函数A执行完,函数B也能访问函数A作用域内的变量,这就是一个闭包——————本质上闭包是将函数内部和外部连接起来的一座桥梁。

  
  
  1. function foo(message) {

  2.    function closure() {

  3.        console.log(message)

  4.    };

  5.    return closure;

  6. }

  7. // 使用

  8. var bar = foo("hello closure!");

  9. bar()// 返回 'hello closure!'

在函数foo内创建的函数closure对象是不能被回收掉的,因为它被全局变量bar引用,处于一直可访问状态。通过执行 bar()可以打印出 hello closure!。如果想释放掉可以将 bar=null即可。

由于闭包会携带包含它的函数的作用域,因此会比其他函数占用更多的内存。过度使用闭包可能会导致内存占用过多。

实例------>demos/closures.html
  
  
  1. <!DOCTYPE html>

  2. <html lang="en">

  3. <head>

  4.  <meta charset="UTF-8">

  5.  <meta name="viewport" content="width=device-width, initial-scale=1.0">

  6.  <meta http-equiv="X-UA-Compatible" content="ie=edge">

  7.  <title>Closure</title>

  8. </head>

  9. <body>

  10.  <p>不断单击【click】按钮</p>

  11.  <button id="click_button">Click</button>

  12.  <script>

  13.    function f() {

  14.      var str = Array(10000).join('#');

  15.      var foo = {

  16.        name: 'foo'

  17.      }

  18.      function unused() {

  19.        var message = 'it is only a test message';

  20.        str = 'unused: ' + str;

  21.      }

  22.      function getData() {

  23.        return 'data';

  24.      }

  25.      return getData;

  26.    }

  27.    var list = [];

  28.    document.querySelector('#click_button').addEventListener('click', function () {

  29.      list.push(f());

  30.    }, false);

  31.  </script>

  32. </body>

  33. </html>

这里结合Chrome的Devtools->Memory工具进行分析,操作步骤如下:

⚠️注:最好在隐藏窗口中进行分析工作,避免浏览器插件影响分析结果

  1. 选中【Record allocation timeline】选项

  2. 执行一次CG

  3. 单击【start】按钮开始记录堆分析

  4. 连续单击【click】按钮十多次

  5. 停止记录堆分析

上图中蓝色柱形条表示随着时间新分配的内存。选中其中某条蓝色柱形条,过滤出对应新分配的对象:

查看对象的详细信息:

从图可知,在返回的闭包作用链(Scopes)中携带有它所在函数的作用域,作用域中还包含一个str字段。而str字段并没有在返回getData()中使用过。为什么会存在在作用域中,按理应该被GC回收掉, why:question:

原因是在相同作用域内创建的多个内部函数对象是共享同一个变量对象(variable object)。如果创建的内部函数没有被其他对象引用,不管内部函数是否引用外部函数的变量和函数,在外部函数执行完,对应变量对象便会被销毁。反之,如果内部函数中存在有对外部函数变量或函数的访问(可以不是被引用的内部函数),并且存在某个或多个内部函数被其他对象引用,那么就会形成闭包,外部函数的变量对象就会存在于闭包函数的作用域链中。这样确保了闭包函数有权访问外部函数的所有变量和函数。了解了问题产生的原因,便可以对症下药了。对代码做如下修改:

  
  
  1.    function f() {

  2.      var str = Array(10000).join('#');

  3.      var foo = {

  4.        name: 'foo'

  5.      }

  6.      function unused() {

  7.        var message = 'it is only a test message';

  8.        // str = 'unused: ' + str; //删除该条语句

  9.      }

  10.      function getData() {

  11.        return 'data';

  12.      }

  13.      return getData;

  14.    }

  15.    var list = [];

  16.    document.querySelector('#click_button').addEventListener('click', function () {

  17.      list.push(f());

  18.    }, false);

getData()和unused()内部函数共享f函数对应的变量对象,因为unused()内部函数访问了f作用域内str变量,所以str字段存在于f变量对象中。加上getData()内部函数被返回,被其他对象引用,形成了闭包,因此对应的f变量对象存在于闭包函数的作用域链中。这里只要将函数unused中 str='unused: '+str;语句删除便可解决问题。

查看一下闭包信息:

DOM泄露

在JavaScript中,DOM操作是非常耗时的。因为JavaScript/ECMAScript引擎独立于渲染引擎,而DOM是位于渲染引擎,相互访问需要消耗一定的资源。如Chrome浏览器中DOM位于WebCore,而JavaScript/ECMAScript位于V8中。假如将JavaScript/ECMAScript、DOM分别想象成两座孤岛,两岛之间通过一座收费桥连接,过桥需要交纳一定“过桥费”。JavaScript/ECMAScript每次访问DOM时,都需要交纳“过桥费”。因此访问DOM次数越多,费用越高,页面性能就会受到很大影响。了解更多ℹ️

为了减少DOM访问次数,一般情况下,当需要多次访问同一个DOM方法或属性时,会将DOM引用缓存到一个局部变量中。但如果在执行某些删除、更新操作后,可能会忘记释放掉代码中对应的DOM引用,这样会造成DOM内存泄露。

实例------>demos/dom.html
  
  
  1. <!DOCTYPE html>

  2. <html lang="en">

  3. <head>

  4.  <meta charset="UTF-8">

  5.  <meta name="viewport" content="width=device-width, initial-scale=1.0">

  6.  <meta http-equiv="X-UA-Compatible" content="ie=edge">

  7.  <title>Dom-Leakage</title>

  8. </head>

  9. <body>

  10.  <input type="button" value="add" class="add">

  11.  <input type="button" value="remove" class="remove" style="display:none;">

  12.  <div class="container">

  13.    <pre class="wrapper"></pre>

  14.  </div>

  15.  <script>

  16.    // 因为要多次用到pre.wrapper、div.container、input.remove、input.add节点,将其缓存到本地变量中

  17.    var wrapper = document.querySelector('.wrapper');

  18.    var container = document.querySelector('.container');

  19.    var removeBtn = document.querySelector('.remove');

  20.    var addBtn = document.querySelector('.add');

  21.    var counter = 0;

  22.    var once = true;

  23.    // 方法

  24.    var hide = function(target){

  25.      target.style.display = 'none';

  26.    }

  27.    var show = function(target){

  28.      target.style.display = 'inline-block';

  29.    }

  30.    // 回调函数

  31.    var removeCallback = function(){

  32.      removeBtn.removeEventListener('click', removeCallback, false);

  33.      addBtn.removeEventListener('click', addCallback, false);

  34.      hide(addBtn);

  35.      hide(removeBtn);

  36.      container.removeChild(wrapper);

  37.    }

  38.    var addCallback = function(){

  39.      wrapper.appendChild(document.createTextNode('\t' + ++counter + ':a new line text\n'));

  40.      // 显示删除操作按钮

  41.      if(once){

  42.        show(removeBtn);

  43.        once = false;

  44.      }

  45.    }

  46.    // 绑定事件

  47.    removeBtn.addEventListener('click', removeCallback, false);

  48.    addBtn.addEventListener('click', addCallback, false);

  49.  </script>

  50. </body>

  51. </html>

这里结合Chrome浏览器的Devtools–>Performance做一些分析,操作步骤如下:

⚠️注:最好在隐藏窗口中进行分析工作,避免浏览器插件影响分析结果

  1. 开启【Performance】项的记录

  2. 执行一次CG,创建基准参考线

  3. 连续单击【add】按钮6次,增加6个文本节点到pre元素中

  4. 单击【remove】按钮,删除刚增加6个文本节点和pre元元素

  5. 执行一次CG

  6. 停止记录堆分析

从分析结果图可知,虽然6次add操作增加6个Node,但是remove操作并没有让Nodes节点数下降,即remove操作失败。尽管还主动执行了一次CG操作,Nodes曲线也没有下降。因此可以断定内存泄露了!那问题来了,如何去查找问题的原因呢?这里可以通过Chrome浏览器的Devtools–>Memory进行诊断分析,执行如下操作步骤:

⚠️注:最好在隐藏窗口中进行分析工作,避免浏览器插件影响分析结果

  1. 选中【Take heap snapshot】选项

  2. 连续单击【add】按钮6次,增加6个文本节点到pre元素中

  3. 单击【Take snapshot】按钮,执行一次堆快照

  4. 单击【remove】按钮,删除刚增加6个文本节点和pre元元素

  5. 单击【Take snapshot】按钮,执行一次堆快照

  6. 选中生成的第二个快照报告,并将视图由"Summary"切换到"Comparison"对比模式,在[class filter]过滤输入框中输入关键字:Detached

从分析结果图可知,导致整个pre元素和6个文本节点无法别回收的原因是:代码中存在全局变量 wrapper对pre元素的引用。知道了产生的问题原因,便可对症下药了。对代码做如下就修改:

  
  
  1.    // 因为要多次用到pre.wrapper、div.container、input.remove、input.add节点,将其缓存到本地变量中

  2.    var wrapper = document.querySelector('.wrapper');

  3.    var container = document.querySelector('.container');

  4.    var removeBtn = document.querySelector('.remove');

  5.    var addBtn = document.querySelector('.add');

  6.    var counter = 0;

  7.    var once = true;

  8.    // 方法

  9.    var hide = function(target){

  10.      target.style.display = 'none';

  11.    }

  12.    var show = function(target){

  13.      target.style.display = 'inline-block';

  14.    }

  15.    // 回调函数

  16.    var removeCallback = function(){

  17.      removeBtn.removeEventListener('click', removeCallback, false);

  18.      addBtn.removeEventListener('click', addCallback, false);

  19.      hide(addBtn);

  20.      hide(removeBtn);

  21.      container.removeChild(wrapper);

  22.      wrapper = null;//在执行删除操作时,将wrapper对pre节点的引用释放掉

  23.    }

  24.    var addCallback = function(){

  25.      wrapper.appendChild(document.createTextNode('\t' + ++counter + ':a new line text\n'));

  26.      // 显示删除操作按钮

  27.      if(once){

  28.        show(removeBtn);

  29.        once = false;

  30.      }

  31.    }

  32.    // 绑定事件

  33.    removeBtn.addEventListener('click', removeCallback, false);

  34.    addBtn.addEventListener('click', addCallback, false);

在执行删除操作时,将wrapper对pre节点的引用释放掉,即在删除逻辑中增加 wrapper=null;语句。再次在Devtools–>Performance中重复上述操作:

小试牛刀------>demos/dom_practice.html

再来看看网上的一个实例,代码如下:

  
  
  1. <!DOCTYPE html>

  2. <html lang="en">

  3. <head>

  4.  <meta charset="UTF-8">

  5.  <meta name="viewport" content="width=device-width, initial-scale=1.0">

  6.  <meta http-equiv="X-UA-Compatible" content="ie=edge">

  7.  <title>Practice</title>

  8. </head>

  9. <body>

  10.  <div id="refA"><ul><li><a href="#"></a></li><li><a href="#"></a></li><li><a href="#" id="refB"></a></li></ul></div>

  11.  <div></div>

  12.  <div></div>

  13.  <script>

  14.    var refA = document.getElementById('refA');

  15.    var refB = document.getElementById('refB');

  16.    document.body.removeChild(refA);

  17.    // #refA不能GC回收,因为存在变量refA对它的引用。将其对#refA引用释放,但还是无法回收#refA。

  18.    refA = null;

  19.    // 还存在变量refB对#refA的间接引用(refB引用了#refB,而#refB属于#refA)。将变量refB对#refB的引用释放,#refA就可以被GC回收。

  20.    refB = null;

  21.  </script>

  22. </body>

  23. </html>

整个过程如下图所演示:

有兴趣的同学可以使用Chrome的Devtools工具,验证一下分析结果,实践很重要~~~:high_brightness:

timers

在JavaScript常用 setInterval()来实现一些动画效果。当然也可以使用链式 setTimeout()调用模式来实现:

  
  
  1. setTimeout(function() {

  2.  // do something. . . .

  3.  setTimeout(arguments.callee, interval);

  4. }, interval);

如果在不需要 setInterval()时,没有通过 clearInterval()方法移除,那么 setInterval()会不停地调用函数,直到调用 clearInterval()或窗口关闭。如果链式 setTimeout()调用模式没有给出终止逻辑,也会一直运行下去。因此再不需要重复定时器时,确保对定时器进行清除,避免占用系统资源。另外,在使用 setInterval()setTimeout()来实现动画时,无法确保定时器按照指定的时间间隔来执行动画。为了能在JavaScript中创建出平滑流畅的动画,浏览器为JavaScript动画添加了一个新API-requestAnimationFrame()。关于setInterval、setTimeout与requestAnimationFrame实现动画上的区别➹猛击

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值