js内存泄漏常见的四种情况
来源:互联网 发布:卖淘宝小号的网站 编辑:程序博客网 时间:2024/06/07 18:10
意外的全局变量
js中如果不用 var
声明变量,该变量将被视为 window
对象(全局对象)的属性,也就是全局变量.
function foo(arg) { bar = "this is a hidden global variable";}// 上面的函数等价于function foo(arg) { window.bar = "this is an explicit global variable";}
所以,你调用完了函数以后,变量仍然存在,导致泄漏.
如果不注意 this
的话,还可能会这么漏:
function foo() { this.variable = "potential accidental global";}// 没有对象调用foo, 也没有给它绑定this, 所以this是windowfoo();
你可以通过加上 'use strict'
启用严格模式来避免这类问题, 严格模式会组织你创建意外的全局变量.
被遗忘的定时器或者回调
var someResource = getData();setInterval(function() { var node = document.getElementById('Node'); if(node) { node.innerHTML = JSON.stringify(someResource)); }}, 1000);
这样的代码很常见, 如果 id
为 Node
的元素从 DOM
中移除, 该定时器仍会存在, 同时, 因为回调函数中包含对 someResource
的引用, 定时器外面的 someResource
也不会被释放.
没有清理的DOM元素引用
var elements = { button: document.getElementById('button'), image: document.getElementById('image'), text: document.getElementById('text')};function doStuff() { image.src = 'http://some.url/image'; button.click(); console.log(text.innerHTML);}function removeButton() { document.body.removeChild(document.getElementById('button')); // 虽然我们用removeChild移除了button, 但是还在elements对象里保存着#button的引用 // 换言之, DOM元素还在内存里面.}
闭包
先看这样一段代码:
var theThing = null;var replaceThing = function () { var someMessage = '123' theThing = { someMethod: function () { console.log(someMessage); } };};
调用 replaceThing
之后, 调用 theThing.someMethod
, 会输出 123
, 基本的闭包, 我想到这里应该不难理解.
解释一下的话, theThing
包含一个 someMethod
方法, 该方法引用了函数中的 someMessage
变量, 所以函数中的 someMessage
变量不会被回收, 调用 someMethod
可以拿到它正确的 console.log
出来.
接下来我这么改一下:
var theThing = null;var replaceThing = function () { var originalThing = theThing; var someMessage = '123' theThing = { longStr: new Array(1000000).join('*'), // 大概占用1MB内存 someMethod: function () { console.log(someMessage); } };};
我们先做一个假设, 如果函数中所有的私有变量, 不管 someMethod
用不用, 都被放进闭包的话, 那么会发生什么呢.
第一次调用 replaceThing
, 闭包中包含 originalThing = null
和 someMessage = '123'
, 我们设函数结束时, theThing
的值为 theThing_1
.
第二次调用 replaceThing
, 如果我们的假设成立, originalThing = theThing_1
和someMessage = '123'
.我们设第二次调用函数结束时, theThing
的值为 theThing_2
.注意, 此时的 originalThing
保存着 theThing_1
, theThing_1
包含着和 theThing_2
截然不同的 someMethod
, theThing_1
的 someMethod
中包含一个 someMessage
, 同样如果我们的假设成立, 第一次的 originalThing = null
应该也在.
所以, 如果我们的假设成立, 第二次调用以后, 内存中有 theThing_1
和 theThing_2
, 因为他们都是靠 longStr
把占用内存撑起来, 所以第二次调用以后, 内存消耗比第一次多1MB.
如果你亲自试了(使用Chrome的Profiles查看每次调用后的内存快照), 会发现我们的假设是不成立的, 浏览器很聪明, 它只会把 someMethod
用到的变量保存下来, 用不到的就不保存了, 这为我们节省了内存.
但如果我们这么写:
var theThing = null;var replaceThing = function () { var originalThing = theThing; var unused = function () { if (originalThing) console.log("hi"); }; var someMessage = '123' theThing = { longStr: new Array(1000000).join('*'), someMethod: function () { console.log(someMessage); } };};
unused
这个函数我们没有用到, 但是它用了 originalThing
变量, 接下来, 如果你一次次调用 replaceThing
, 你会看到内存1MB 1MB的涨.
也就是说, 虽然我们没有使用 unused
, 但是因为它使用了 originalThing
, 使得它也被放进闭包了, 内存漏了.
强烈建议读者亲自试试在这几种情况下产生的内存变化.
这种情况产生的原因, 通俗讲, 是因为无论 someMethod
还是 unused
, 他们其中所需要用到的在 replaceThing
中定义的变量是保存在一起的, 所以就漏了.
如果我没有说明第四种情况, 可以参考以下链接, 或是在评论区评论.
参考链接
An interesting kind of JavaScript memory leak
一个意想不到的Javascript内存泄漏
Grokking V8 closures for fun (and profit?)
- js内存泄漏常见的四种情况
- js内存泄漏常见的四种情况
- js内存泄漏常见的四种情况
- js内存泄漏常见的四种情况
- js内存泄漏常见情况
- js内存泄漏的几种情况
- iOS内存泄漏的常见情况
- 导致JS内存泄漏的几种情况
- java内存泄漏常见情况
- 内存泄漏的情况
- Android开发中常见的内存泄漏情况和解决办法
- 内存泄漏的几种情况
- 内存泄漏的几种情况
- 内存泄漏的几种情况
- 导致内存泄漏的八种情况
- C++内存泄漏的几种情况
- JavaScript内存管理机制以及四种常见的内存泄漏解析
- JavaScript内存管理机制以及四种常见的内存泄漏解析
- hrbustoj.G.FBI Tree(2016级新生程序设计全国邀请赛 )
- git安装
- MySQL数据库5.X版本基本手工注入总结
- UVa 156
- mui:文件上传
- js内存泄漏常见的四种情况
- spring 源码探索--单例bean解决循环依赖问题
- Android视频框架 Vitamio 打造自己的万能播放器
- C++中substr函数的用法
- BITCS2016程序设计 | 21. 穿越矩阵
- NOIP2016 提高组 爆炸记
- 快速幂取模
- 《失控》讨论会感触
- 视频框架Vitamio学习