垃圾回收
垃圾回收会定期找出那些不继续使用的变量,然后释放其内存
垃圾回收器会按照固定的时间间隔周期性的执行
只有函数内的变量才可能被回收
function fn1() {
var obj = {name: 'hanzichi', age: 10};
}
function fn2() {
var obj = {name:'hanzichi', age: 10};
return obj;
}
// 而当调用结束后,出了fn1的环境,那么该块内存会被自动释放
var a = fn1();
var b = fn2();
哪个变量是没有用的
通常情况下有两种实现方式:标记清除和引用计数
标记清除是较为常用的
而引用计数不是很常用
标记清除
js中最常用的垃圾回收方式就是标记清除
例如,在函数中声明一个变量,就将这个变量标记为"进入环境"。
而当变量离开环境时,则将其标记为"离开环境"。
function test(){
var a = 10 ; // 被标记 ,进入环境
var b = 20 ; // 被标记 ,进入环境
}
// 只有没有使用到的数据才会标记为离开环境 被回收
test(); // 执行完毕 之后 a、b又被离开环境,被回收
引用计数
引用计数的含义是跟踪记录每个值被引用的次数。
当声明了一个变量并将一个引用类型值赋给该变量时,则这个值的引用次数就是1。
如果同一个值又被赋给另一个变量,则该值的引用次数加 1。
相反,如果包含对这个值引用的变量又取得了另外一个值,则这个值的引用次数减 1
function test() {
var a = {}; // a指向对象的引用次数为1
var b = a; // a指向对象的引用次数加1,为2
var c = a; // a指向对象的引用次数再加1,为3
var b = {}; // a指向对象的引用次数减1,为2
}
// 如果最后引用次数为0时,就会被清除掉
内存泄漏
不再用到的内存,没有及时释放,就叫做内存泄漏
1.循环引用
一个很简单的例子:一个DOM对象被一个Javascript对象引用
与此同时又引用同一个或其它的Javascript对象,这个DOM对象可能会引发内存泄露。
这个DOM对象的引用将不会在脚本停止的时候被垃圾回收器回收。要想破坏循环引用,引用DOM元素的对象或DOM对象的引用需要被赋值为null
2.闭包
// 闭包可以维持函数内局部变量,使其得不到释放
在闭包中引入闭包外部的变量时,当闭包结束时此对象无法被垃圾回收(GC)
3.DOM泄露
当原有的DOM被移除时,子结点引用没有被移除则无法回收
4.Times计时器泄露
定时器忘记关闭,一直就不会被清除
意外的全局变量
function foo(){
// bar是全局变量 这种变量一直不会被清除
bar = "this is a hidden global variable";
}