目录
一、javascript垃圾回收机制
标记清除
引用计数
二、减少JavaScript中的垃圾回收
对象优化
数组优化
函数优化
高级技术
解决内存的泄露,垃圾回收机制会定期(周期性)找出那些不再用到的内存(变量),然后释放其内存。
现在各大浏览器通常采用的垃圾回收机制有两种方法:标记清除,引用计数。
js中最常用的垃圾回收方式就是标记清除。垃圾回收机制在运行的时候会给存储再内存中的所有变量都加上标记(可以是任何标记方式),然后,它会去掉处在环境中的变量及被环境中的变量引用的变量标记(闭包)。而在此之后剩下的带有标记的变量被视为准备删除的变量,原因是环境中的变量已经无法访问到这些变量了。最后垃圾回收机制到下一个周期运行时,将释放这些变量的内存,回收它们所占用的空间。
function test(){
var a = "world; //被标记"进入环境"
var b = "hello"; //被标记"进入环境"
}
test(); //执行完毕后之后,a和b又被标记"离开环境",被回收
工作原理:
当变量进入环境时,将这个变量标记为“进入环境”。当变量离开环境时,则将其标记为“离开环境”。标记“离开环境”的就回收内存。
工作流程:
语言引擎有一张"引用表",保存了内存里面所有资源(通常是各种值)的引用次数。如果一个值的引用次数是0,就表示这个值不再用到了,因此可以将这块内存释放。
上图中,左下角的两个值,没有任何引用,所以可以释放。
如果一个值不再需要了,引用数却不为0,垃圾回收机制无法释放这块内存,从而导致内存泄漏。
工作原理:跟踪记录每个值被引用的次数。
工作流程:
;
const arr = [9,8,7,6];
console.log("hello world")
上面的代码中,数组[9,8,7,6]是一个值,会占用内存。变量arr是仅有的对这个值的引用,因此引用次数为1。尽管后面的代码没有用到arr,它是会持续占用内存。
如果增加一行代码,解除arr对[9,8,7,6]引用,这块内存就可以被垃圾回收机制释放了。
let arr = [9,8,7,6];
console.log("hello world");
arr = null;
上面代码中,arr重置为null,就解除了对[9,8,7,6]的引用,引用次数变成了0,内存就可以释放出来了。
但是循环引用的时候就会释放不掉内存——造成内存泄漏
因为IE中的BOM、DOM的实现使用了COM,而COM对象使用的垃圾收集机制是引用技术策略。所以会存在循环引用的问题。
解决:手工断开JS对象和DOM之间的链接,赋值为null。IE9把DOM和BOM转换成真正的JS对象了,所以避免了这个问题。
首先,最明显的,new关键字就意味着一次内存分配,例如 new Foo()。最好的处理方法是:在初始化的时候新建对象,然后在后续过程中尽量多的重用这些创建好的对象。
另外还有以下三种内存分配表达式(可能不像new关键字那么明显了):
在浏览器开发过程中,随着内存对象的不断创建,浏览器会执行间歇性的内存清理,在对象创建非常多的情况下,会出现体验非常卡的现象。
对象的创建的形式
{}
[]
function (){}
常用在函数的返回值是一个对象,使用全局对象(确保对象prototype上没有属性),每次擦出对象上的属性,反复利用。
cr.wipe = function(obj) {
var p;
for(p in obj) {
if(obj.hasOwnProperty(p)) {
delete obj[p];
}
}
}
将[]赋值给一个数组对象,能够达到清空数组的目的,但会创建一个新的数组对象,原来的将会占用一小片内存。可以使用arr.length=0也达到清空数组,实现数组重复利用,达到减少内存的产生。
在使用定时器的函数中,每次执行都会创建新函数,如下形式
setTimeout(
(function(self) {
return function() {
self.tick();
}
})(this), 30);
每一调用都会产生一个新的方法对象,导致大量垃圾对象的产生。为解决这个问题,可以讲返回值作为对象保存起来。
this.tickFunc = (
function(self) {
return function() {
self.tick();
}
}
)(this);
setTimeout(this.tickFunc, 30);
从根本上来说,javascript本身就是围绕着垃圾收集来设计的。随着我们工作的进行,避免内存垃圾变 得越来越困难。因为很多方便实用的Javascript库方法也会产生一些新的对象。对于这些库方法产生的垃圾,我们束手无策,只能重新翻看文档,并且检 查方法的返回值。例如,数组的slice方法返回一个新的数组(在不修改原数组的基础上,截取出一部分作为新数组),字符串的substr方法返回一个新 的字符串(在不修改原字符串的基础上,截取出一部分字符串作为返回值)等等。
调用这些库方法,将会创建内存垃圾,而你能做的,只有避免调用这些方法,或者用不创建系统垃圾的方式重写这些方法(有点极端啦~)。
例如,在Construct 2引擎中,从数组中利用下标来删除一个元素,是经常进行的操作。最初我们是用下面这种方式来实现的:
var sliced = arr.slice(index + 1);
arr.length = index;
arr.push.apply(arr, sliced);
然而,slice方法会返回一个新的数组对象(数组中的元素是原数组中删掉的部分),并且会通过 arr.push.apply方法将元素重新复制回原数组,但是在此操作之后,该数组就成为了一片内存垃圾。由于这是我们引擎中的垃圾产生的热点代码(使 用频率非常很高),因此我们利用了迭代的方式重写了上述代码:
for (var i = index, len = arr.length – 1; i < len; i++)
arr[i] = arr[i + 1];
arr.length = len;
显然,重写大量的库函数是非常痛苦的,因此你必须仔细权衡方法的易用性和内存垃圾产生情况。如果产生大量内存垃圾的方法在动画的每一帧中被多次调用,你可能就会兴高采烈的重写库函数啦。
在递归函数中,通过{}构造空对象,并在递归过程中传递数据,虽然是很方便的。但是更好的方式是:利用一个 单独的数组对象作为堆栈,在递归过程中对数组进行push和pop操作。更进一步,不要调用array的pop方法(pop将会使得array的最后一个 元素将会变成内存垃圾),而应该使用一个索引来记录数组的最后一个元素的位置,在pop时简单的将索引减一即可;类似的,将索引加1来代替array的 push操作,只有当索引对应的元素不存在时,才执行真正的push为数组加入一个新元素。
另外,在任何时候,都应该避免使用向量对象(例如:包含x和y属性的vector2对象)。有些方法将向量 对象作为方法返回值,既可以支持返回值的再次修改,又能够将需要的属性一次性返回,使用起来非常方便。但是有时候在一帧动画中,创建了成百上千个这样的向 量对象,从而导致严重的垃圾回收性能问题,也是非常常见的。因此最好将这些方法分离成具有独立职责的功能个体,例如:利用getX()和getY()方法 (返回具体数据)代替getPosition()方法(返回一个vector2对象)
有时候,有的库是一个生产垃圾的噩梦,人人趋而避之,可是你或许就偏偏强烈依赖于这样的库。 Box2Dweb就是一个典型的例子:这个库在每一帧都会产生成百上千个b2Vec2对象,持续向浏览器中注入内存垃圾,最终导致严重的垃圾收集暂停。针 对这种情况,最好的解决办法就是创建一个重复利用的对象缓存,我们正在对一个修改后版本的Box2D进行测试,这个版本已经创建了对象缓存,并且它看上去 能够有助于缓解(虽然并没有完全解决)垃圾收集暂停。Get和Free的源码请参见b2Vec2.js。
新版的Box2D中存在一个被称为“自由缓存”(free cache)的数组,在任何涉及b2Vec2对象操作的地方都包含了对自由缓存的考虑。如果b2Vec2对象不再使用,则将此对象放置在自由缓存中;当需 要新建b2Vec2对象是,如果自由缓存中存在对象,则重用这些对象,如果不存在,才会创建一个新的对象。这种方案并不完美,因为在我进行的一些测试中, 只有一半的b2Vec2对象能被重复利用,但是这种方案,的的确确减少了垃圾回收的压力,并且有效的减少了垃圾回收暂停的频率。
在Javascript中,彻底避免垃圾回收是非常困难的。垃圾回收机制与实时软件(例如:游戏)的实时性要求,从根本上就是对立的。
但是,为了减少内存垃圾,我们还是可以对javascript代码进行彻底检查,有些代码中存在明显的产生过多内存垃圾的问题代码,这些正是我们需要检查并且完善的。