古老定义
-
闭包(closure),是指函数变量可以保存在函数作用域内,因此看起来是函数将变量“包裹”了起来。
-
//根据定义,包含变量的函数就是闭包
-
function foo() {
-
var a = 0;
-
}
-
cosole.log(a)
-
// Uncaught ReferenceError: a is not defined
-
《JavaScript高级程序设计》对闭包定义
闭包是指有权访问另一个函数作用域中的变量的函数
-
//根据《JavaScript高级程序设计》,访问上层函数的作用域的内层函数就是闭包
-
function foo() {
-
var a = 2;
-
function bar() {
-
console.log(a);
-
}
-
bar();
-
}
-
foo();
《JavaScript权威指南》对闭包定义
函数对象可以通过作用域链相互关联起来,函数体内部变量可以保存在函数作用域内,这就是闭包。
-
var global = "global scope"; //全局变量
-
function checkscope() {
-
var scope = "local scope"; //局部变量
-
function f() {
-
return scope; //在作用域中返回这个值
-
};
-
return f();
-
}
-
checkscope(); // 返回 "local scope"
严格来说,闭包需要满足三个条件:【1】访问所在作用域;【2】函数嵌套;【3】在所在作用域外被调用
有些人觉得只满足条件1就可以,所以IIFE是闭包;有些人觉得满足条件1和2才可以,所以被嵌套的函数才是闭包;有些人觉得3个条件都满足才可以,所以在作用域以外的地方被调用的函数才是闭包
为什么我们需要闭包
首先来看一个例子,我们来实现一个计数器。
-
var counter = 0;
-
function add() {
-
return counter += 1;
-
}
-
add();
-
add();
-
add();// 计数器现在为 3
现在我们已经达到了目的,可是问题来了,代码中的任何一个函数都可以随意改变counter的值,所以这个计数器并不完美。那我们把counter放在add函数里面不就好了么?
-
function add() {
-
var counter = 0;
-
return counter += 1;
-
}
-
add();
-
add();
-
add();// 本意是想输出 3, 但输出的都是 1
所以这样做的话,每次调用add函数,counter的值都要被初始化为0,还是达不到我们的目的。
如何使用闭包
所以这时候我们就要用闭包去解决这个问题了,先看代码。
-
var add = (function () {
-
var counter = 0;
-
return function () {return counter += 1;}
-
})();
-
add();
-
add();
-
add();// 计数器为 3
这时候我们完美实现了计数器。这段非常精简,可以拆分成如下等价代码。
-
function outerFunction () {
-
var counter = 0;
-
function innerFunction (){
-
return counter += 1;
-
}
-
return innerFunction;
-
}
-
var add = outerFunction();
-
add();
-
add();
-
add();// 计数器为 3
这时候的add就形成了一个闭包。一个闭包由两部分组成,函数和创建该函数的环境。环境是由环境中的局部变量组成的。对于闭包add来说,它由函数innerFunction和变量counter组成,所以这时候add是可以访问变量counter的。
使用闭包应注意的问题
由于闭包会携带包含它的函数的作用域,因此会比其他函数占用更多的内存。因此可以手动解除对匿名函数的引用,以便释放内存。
-
function f2(){
-
var n=22;
-
var nAdd=function(){n++};
-
return function(){
-
return {
-
n:n,
-
nAdd:nAdd
-
}
-
}
-
}
-
//result2就是创建了一个匿名函数
-
var result2=f2();
-
//调用函数
-
console.log(result2());
-
result2().nAdd();
-
console.log(result2());
-
//解除对匿名函数的引用,以便释放内存
-
result2=null;