[摘译]js内存泄漏常见的四种情况
本文主要选取了4 Types of Memory Leaks in JavaScript and How to Get Rid Of Them 这篇文章中的一小部分来说明一下js中产生内存泄漏的常见情况. 对于较难理解的第四种情况, 参考了一些文章来进行说明.
意外的全局变量
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是window
foo();
你可以通过加上'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内存泄漏常见的四种情况(From LeuisKen)
本文主要选取了4 Types of Memory Leaks in JavaScript and How to Get Rid Of Them 这篇文章中的一小部分来说明一下js中产生内存泄漏的常见情 ...
- 四种情况会造成内存的泄漏:
以下四种情况会造成内存的泄漏: 意外的全局变量: 由于使用未声明的变量,而意外的创建了一个全局变量,而使这个变量一直留在内存中无法被回收. 被遗忘的计时器或回调函数: 设置了 setInterval ...
- JS内存泄漏排查方法——Chrome Profiles
本文转自JS内存泄漏排查方法--Chrome Profiles 概述 Google Chrome浏览器提供了非常强大的JS调试工具,Heap Profiling便是其中一个.Heap Profilin ...
- js和php能生成一样的随机数_JavaScript_JS生成某个范围的随机数【四种情况详解】,前言:
JS没有现成的函数,能 - phpStudy...
JS生成某个范围的随机数[四种情况详解] 前言: JS没有现成的函数,能够直接生成指定范围的随机数. 但是它有个函数:Math.random() 这个函数可以生成 [0,1) 的一个随机数. 利用它 ...
- 对Java中常见的四种I/O模型理解
对Java中常见的四种I/O模型理解 1.1 知识科普 1.1.1 同步阻塞I/O(Blocking I/O) 1.1.1.1 阻塞与非阻塞 1.1.1.2 同步与异步 1.1.1.3 同步阻塞I/O ...
- 【JS继承】常见的7种继承方式
自我介绍:大家好,我是吉帅振的网络日志:微信公众号:吉帅振的网络日志:前端开发工程师,工作4年,去过上海.北京,经历创业公司,进过大厂,现在郑州敲代码. JS继承专栏 1[JS继承]什么是JS继承? ...
- 内存泄漏常见场景及处理方法
1.Java垃圾回收(GC) Java 是如何管理内存 为了判断Java中是否有内存泄露,我们首先必须了解Java是如何管理内存的.Java的内存管理就是对象的分配和释放问题.在Java中,程序员需要 ...
- post 表单中常见的四种表单请求方式
//post 表单中常见的四种表单请求方式 typedef enum { postformURLEncoded, /*对应Content-Type: appli ...
- js的event loop/js内存泄漏
js的event-loop机制 event-loop主要有三部分组成 执行栈,消息队列.微任务队列 执行优先级是 执行栈>微任务队列>消息对系列 js是单线程语言,event loop开始 ...
最新文章
- golang1.16新特性速览
- 使用Nessus漏扫
- Apollo配置灰度发布
- Mybatis中使用Dao实现类实现增删改查【实际开发中使用代理dao】
- linux gunzip r .tar,Linux命令之文件归档、压缩、解压缩tar,gzip,gunzip,bzip2,bunz -电脑资料...
- gym 101657 D
- 郝斌数据结构教学视频下载地址
- ios 请在设置中打开相机权限_ios开发相机权限问题
- hp linux 禁用u盘启动项,惠普台式机UEFI BIOS设置U盘启动
- html5课件动画制作,ppt如何制作课件动画
- 什么是源路由(source routing)?
- HDOJ 5773 The All-purpose Zero
- 《程序员》杂志 · 2017 精华本
- MTK 连接CCT 连不上 出现CDC Serial驱动无法识别
- jar包太大?手把手教你分析 Maven 依赖,完成瘦身优化!
- Android cer转bks
- selenium设置浏览器的大小和位置
- 炫龙T3-pro 9代cpu无csm兼容选项笔记本GPT硬盘纯uefi安装windows7系统方法
- java定时任务cron表达式每周执行一次的坑
- 遥控小车(基于TCP/IP)