闭包就是就是函数的“堆栈”在函数返回后并不释放,我们也可以理解为这些函数堆栈并不在栈上分配而是在堆上分配。
当在一个函数内定义另外一个函数就会产生闭包。
上面的第二定义是第一个补充说明,抽取第一个定义的主谓宾——闭包是函数的‘局部变量’集合。只是这个局部变量是可以在函数返回后被访问。
我的理解是,闭包就是能够读取其他函数内部变量的函数。
由于在Javascript语言中,只有函数内部的子函数才能读取局部变量,因此可以把闭包简单理解成"定义在一个函数内部的函数"。
所以,在本质上,闭包就是将函数内部和函数外部连接起来的一座桥梁。
例如:
function f1(){
var n=999;
function f2(){
alert(n);
}
return f2;
}
var result=f1();
result(); // 999
明白闭包的含义,再来看看闭包的特点,如下:
1.闭包中局部变量是引用而非拷贝
例如:
function sayNum() {
var num = 2013;
var sayAlert = function() { alert(num); }
num++;
return sayAlert;
}
var sayAlert = sayNum();
sayAlert();//执行结果2014
2.一个函数内可以同时拥有多个内置函数,形成一个含有多个函数的闭包
例如:
functionf1() {
var num = 123;
gAlertNumber = function() { alert(num); }
gIncreaseNumber = function() { num++; }
gSetNumber = function(x) { num = x; }
}
f1(); // 为三个全局变量赋值
gAlertNumber();//123
gIncreaseNumber();
gAlertNumber();//124
gSetNumber(128);
gAlertNumber();//128
3.外部函数所有局部变量都在闭包内,即使这个变量声明在内部函数定义之后
例如:
function sayAlice() {
var sayAlert = function() { alert(alice); }
var alice = 'Hello Alice';
return sayAlert;
}
var helloAlice=sayAlice();
helloAlice();//执行结果是弹出”Hello Alice”的窗口
最后,总结下闭包使用的注意点:
1.由于闭包会使得函数中的变量都被保存在内存中,内存消耗很大,所以不能滥用闭包,否则会造成网页的性能问题,在IE中可能导致内存泄露。解决方法是,在退出函数之前,将不使用的局部变量全部删除。
2.闭包会在父函数外部,改变父函数内部变量的值。所以,如果你把父函数当作对象(object)使用,把闭包当作它的公用方法(Public Method),把内部变量当作它的私有属性(private value),这时一定要小心,不要随便改变父函数内部变量的值。