javascript 闭包是如何处理父函数中 let 形成的块级作用域中的变量呢?
function test(){
var arr = [];
for(var i = 0;i < 10;i++){
arr[i] = function(){
return i;
};
}
for(var a = 0;a < 10;a++){
console.log(arr[a]());
}
}
test(); // 连续打印 10 个 10
为什么把上面第三行代码的 var i = 0 改为 let i = 0 后,调用 test() ,就会打印 1到9 了呢?
javascript 闭包是如何处理父函数中 let 形成的块级作用域中的变量呢?
varr的时候,如下:
函数1作用域
for(var i = 0; i < 10; i++) { 函数1作用域
我在函数1作用域中
arr[i] = function() { 函数2作用域
我在函数2作用域中
return i;
};
}
函数1作用域
console.log(i); 毫无疑问,执行到这里的时候,i是10,既然这里是10
那么在函数2作用域中访问i也是10也就不足为奇了
因为函数2作用域中没有,向上去函数1作用域中找
同一作用域中同一变量名的变量值肯定是相同的(未修改的情况下)
--------------------------分界线------------------------- 当你用 let 的时候,如下:
块1作用域
for(let i = 0; i < 10; i++) { 块2作用域
我在块2作用域中
console.log(i); // 毫无疑问,这里的i从0依次增加到10
arr[i] = function() { 块3作用域
我在块3作用域中
return i;
};
}
块1作用域
当你换成let的时候,读取i的时候,在当前作用域(块3)中没有找到,向上一个作用域(块2)寻找,在块2中发现i,于是拿到值。
- ==闭包中的 this 对象==
var name = "The Window";
var obj = {
name: "My Object",
getName: function(){
return function(){
return this.name;
};
}
};
console.log(obj.getName()()); // The Window
obj.getName()()实际上是在全局作用域中调用了匿名函数,this指向了window。这里要理解函数名与函数功能(或者称函数值)是分割开的,不要认为函数在哪里,其内部的this就指向哪里。匿名函数的执行环境具有全局性,因此其 this 对象通常指向 window。
var name = "The Window";
var obj = {
name: "My Object",
getName: function(){
var that = this;
return function(){
return that.name;
};
}
};
console.log(obj.getName()()); // My Object
- 匿名函数最大的用途是创建闭包,并且还可以构建命名空间,以减少全局变量的使用。从而使用闭包模块化代码,减少全局变量的污染。
var objEvent = objEvent || {};
(function(){
var addEvent = function(){
// some code
};
function removeEvent(){
// some code
}
objEvent.addEvent = addEvent;
objEvent.removeEvent = removeEvent;
})();
在这段代码中函数 addEvent 和 removeEvent 都是局部变量,但我们可以通过全局变量 objEvent 使用它,这就大大减少了全局变量的使用,增强了网页的安全性。
- 一个闭包计数器
var countNumber = (function(){
var num = 0;
return function(){
return ++num;
};
})();
闭包的缺陷
- 闭包的缺点就是常驻内存会增大内存使用量,并且使用不当很容易造成内存泄露。
- 如果不是因为某些特殊任务而需要闭包,在没有必要的情况下,在其它函数中创建函数是不明智的,因为闭包对脚本性能具有负面影响,包括处理速度和内存消耗。
最后再来一些有关闭包的面试题
- 下面代码中,标记 ? 的地方输出分别是什么?
function fun(n,o){
console.log(o);
return {
fun: function(m){
return fun(m,n);
}
};
}
var a = fun(0); // ?
a.fun(1); // ?
a.fun(2); // ?
a.fun(3); // ?
var b = fun(0).fun(1).fun(2).fun(3); // ?
var c = fun(0).fun(1); // ?
c.fun(2); // ?
c.fun(3); // ?
undefined
0
0
0
undefined, 0, 1, 2
undefined, 0
1
1
参考资料
- 【书】《JavaScript 高级程序设计(第三版)》
- 【文章】学习Javascript闭包(Closure)
- 【文章】【JavaScript】【函数】闭包闭包!