一、什么是闭包
- 首先闭包一定是函数对象。
- 函数内保持对上层作用域的引用。
- 闭包和词法作用域、作用域链、垃圾回收机制等息息相关。
- 当函数在其定义的作用域外进行访问时,才会产生闭包。
- 闭包是由该函数和其上层执行上下文共同构成。
二、闭包的作用
- 在函数外读取函数内部的变量。
- 让局部变量的值能够被保存下来。
- 将模块的公有属性和方法暴露出来。
举个例子:
function fn(){
var i = 5;
return function f2(){
i++;
return i;
}
}
var con = fn();
con();
三、作用域的概念
词法作用域:也叫静态作用域,它的作用域是指在词法分析阶段就确定了,不会改变。
举个例子:
var n = 1;
function f1() {
console.log(n) // 结果为1
}
function f2(){
var n = 2;
f1();
}
f2();
动态作用域:是在运行时根据程序的流程信息来动态确定的,而不是在写代码时进行静态确定的。
区别:词法作用域关注函数在何处声明,而动态作用域关注函数从何处调用。
作用域链:本质上是一个指向变量对象的指针列表,它只引用但不实际包含变量对象。
举个例子:
function compare(value1,value2){
if(value1 < value2){
return -1;
} else if( value1 > value2 ) {
return 1;
} else {
return 0;
}
}
var result = compare(5, 10);
console.log(n) // 结果为-1
四、垃圾回收机制
各种浏览器通常采用的垃圾回收有两种方法:标记清除、引用计数
标记清除:
当变量进入执行环境时,将这个变量标记为“进入环境”。当变量离开执行环境时,则将其标记为“离开环境”,就销毁回收内存。
引用计数:
跟踪记录每个值被引用的次数,当引用次数变成0时,就销毁回收内存
举个例子:
function fn1(){
var n = 5;
n++;
return n;
}
console.log( fn1() ); // 结果为6
五、闭包优缺点
优点:可以防止对全局作用域的污染。
缺点:闭包会使得函数中的变量被保存在内存中,增加内存消耗,如果滥用闭包是话,可能会造成网页的性能问题,在低版本IE中还可能导致内存泄露。