Javascript中常见的内存泄漏

内存泄漏:内存泄漏指由于错误或疏忽导致程序未能释放已经不再使用的内存,内存泄漏并非内存在物理上的消失,而是应用程序分配某段内存后,由于设计错误,导致在释放该内存之前就失去了对这段内存的控制,从而造成了内存的浪费。--维基百科

下面整理出几种常见的引起内存泄漏的情况

意外的全局变量

当在函数中声明变量时忘记使用var(或let、const)声明变量,就会意外地声明一个全局变量,这个全局变量实际上相当于变成了全局对象window的一个属性:

function foo() {
     bar = "Million";
}
foo()       

正常来讲,一个函数执行完成后,函数中不再被使用到的变量就会被全部销毁,由于bar是一个全局变量,所以bar并不会被销毁,会一直保留在内存中,只有页面被关闭或关闭浏览器才会被释放

还有通过this意外创建的全局变量:

function foo() {
    this.bar = "Million";
}
foo()

函数在全局环境执行时,this指向的就是window对象,相当于另一种方式给window对象增加了属性。

如果想避免这两种情况,可以使用‘use strict’严格模式,严格模式下的this不会自动指向window,而是指向undefined. 同时严格模式下禁止在未声明变量的情况下就使用变量。 ( var 就是变量声明操作符 )

console.log

console.log: 向Web控制台打印一条消息,常用来在开发时调试分析,有时会在开发过程中打印出对象信息,但是在网站发布时忘记删除,这个时候,为了维持对象可以再控制台打印,该对象就会被一直留在对象中。这个对象将无法被垃圾回收。

闭包

在同一个外部函数下,如果有多个内部函数,这些内部函数是共享同一个外部函数对象的,如果创建的内部函数没有被外部变量引用,无论内部函数是否有使用外部函数的变量,在函数执行完后,外部函数和和内部函数的变量都将被销毁,反之,如果有一个或多个内部函数被外部变量引用,且内部函数有使用到外部函数的变量,就形成了闭包,外部函数的变量就会存在于内部函数的作用域链中,哪怕这个内部函数没有使用到这个外部函数内变量,但只要同一级的其他内部函数使用到了这个外部函数内变量,这个变量就会存在所有会被外部变量引用的内部函数中。

DOM泄漏

我们知道由于浏览器的原因,当使用JS操作DOM时都会耗费很高的性能,因为DOM是放在浏览器的WebCore中,而JS的操作是放在V8引擎中,JS要取到DOM就相当于要跨过一条长长的桥,需要大量性能。但这不是引起内存泄漏的原因。原因是,我们为了减少性能损失,会在JS中,对DOM的引用放在一个变量中,避免每次都重复引用相同的变量多次,但很多时候,我们引用完DOM后都不会把该变量清除(null 或者 undefined),导致该变量一直保持对DOM的引用,造成内存泄漏。

有一种特殊情况是即使将变量设null也无法让比变量被GC的:

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

//js
var refA = document.getElementById('refA');
var refB = document.getElementById('refB');

document.body.removeChild(refA);

refA = null;  //此时由于refB还留有对refA中孙元素refB的引用,所以refA还无法被GC
refB = null; //refB置空后没有变量保持对refeA的引用,refA被GC
image.png

timers

使用setInterval定时器时如果没有clearInterval,定时器就会一直调用。即使你把调用定时器的那个对象置null,这实际上也是一种闭包。

EventListener

做移动端适配时,需要对不同尺寸设备做适配,当需要考虑横竖屏适配时,一般是在横屏发生变化时,需要将组件销毁再重新生成,而在组件中会对其相关事件绑定,如果在销毁组件时,没有将组件的事件解绑,在横竖屏发生变化时,就会不断地对组件进行事件绑定,这样会导致一些异常。甚至页面崩溃。

这篇文章其实是 常见的 JavaScript 内存泄露
的阅读笔记,里面用Chrome控制台的Performance和Memory演示了查看内存泄漏,我觉得是本文最有趣的地方。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,033评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,725评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,473评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,846评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,848评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,691评论 1 282
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,053评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,700评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,856评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,676评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,787评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,430评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,034评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,990评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,218评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,174评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,526评论 2 343