http://www.cnblogs.com/yunfeifei/p/4019504.html
闭包(closure)简单的说,Javascript允许使用内部函数---即函数定义和函数表达式位于另一个函数的函数体内。而且,这些内部函数可以访问它们所在的外部函数中声明的所有局部变量、参数和声明的其他内部函数。当其中一个这样的内部函数在包含它们的外部函数之外被调用时,就会形成闭包。
几个关键字 函数内的函数 外部函数之外调用
嵌套函数的作用域特点函数 B 下面又包含了函数 C。此时函数 C 为函数 B 的嵌套函数,函数 B 为函数 C 的容器函数。对于C来说也具有刚刚讲过的 “你的就是我的,我的还是我的” 的特点。以此类推层层嵌套的话就形成了一条链条, 作用域按此规律也形成了 Javascript 中的作用域链。
更理论一点:
当一个函数被创建时,同时创建了一个闭包(其中包括:该函数,以及在此时可以访问到的所有变量)
具体实现:
函数运行上下文、词法环境
1、函数运行上下文(每一次函数调用,都会创建一个新得函数运行上下文【调用结束后,后被抛弃】)
程序运行:后形成一个有“函数运行上下文”构成的栈
2、词法环境(运行环境的一个成分)
代码片段(代码块({}形成的),函数)
代码片段每一次运行,都会创建一个新的词法环境
在代码片段的运行中,当遇到一个标识符,就尝试通过词法环境获得这个值
当赋值时,就在词法环境更新这个值
当创建一个函数是,创建这个函数的代码片段所关联的词法环境,会被存放在函数的内部属性[[Environment]]
所以:当一个函数被调用时
1)函数的运行上下文创建,并压入栈
2)创建对应的词法环境,并与上下文关联
3)将外层词法环境设置为该函数的[[Environment]]值
(function(){})()
2)、结果缓存
function Person(){
let name='dede';//私有变量
return {//public
getName:function(){
return name
},
setName(_name){
name=_name
}
}
}
let p = new Person()
p.setName("Tom");
console.log(p.getName())
闭包的注意点
function A(){
var num = 6;//外部的名为num 的变量
function B(num){
return num;//当做参数传进来的num 变量,命名冲突发生在这
}
return B;
}
var result = A()(10);
console.log(result);//输出结果10
一些常见题:
function createFunctions(){
var result = new Array();
for (var i=0; i < 10; i++){
result[i] = function(){
return i;
};
}///这个地方没有(),所以不会执行函数,但是i会被保留下来,为10
return result;
}
var funcs = createFunctions();//返回result,为null
for (var i=0; i < funcs.length; i++){
console.log(funcs[i]());/////funcs[i]()==》result[i](),这个地方才会执行,但是i为10
}
2)
function A(a){
function B(b){
return a + b;
}
return B;
}
var C = A(1);//a被保存了起来 C=B,这是B不执行 [Function B]
//a何时被销毁,当 B 没有再被引用的时候,就会被销毁.
var result = C(2);//b C()=>B()=>执行函数
console.log(result);//输出结果 3