刷牛客网编程题的时候遇到使用闭包的题,以前总觉得自己似乎知道,但真正用起来的时候才发现并不真正理解。去翻了js高级程序设计,7.2节闭包,同时提到了匿名函数,又上网查,最终整理到这里,也算是一次梳理。
一、首先看一个经典例题。
var result = new Array();
function func(){
for(var i=0;i<10;i++){
result[i] = function(){
return i;
}
}
}
console.log( result[0]()); //10
返回的函数数组result[i]执行的结果都是10,这是因为(重点!!!!)js在函数执行前,都只对变量保持引用,并不会真正获取和保存变量的值。而当循环结束后,变量i的值已经变成了10,此时每个函数都引用着i的同一个变量对象,所以每个函数内部i都是10。此时由于return[0]到return[9]保存的都是返回的内部函数,可以访问外部函数的变量i,也就是说形成了闭包。
闭包指的就是有权访问另一个函数作用域中变量的函数,常见的创建闭包的方式就是在一个函数内部创建一个内部函数。
另外,某个执行环境中的代码被执行完毕后,该环境被销毁,保存在该执行环境中的变量和函数也会被销毁。但在这个时候,i依然被result函数引用,仍然占用内存。
我们想要的是输出0到9,怎么解决呢?
1.第一个方法是采用创建匿名函数的方法使得闭包符合预期。是《js高级程序设计》里的解决方法。在内部函数(也就是闭包)外面加上一层匿名函数:
var result = new Array();
function func(){
for(var i=0;i<10;i++){
result[i] = function(num){ //返回给result[i]的仍然是函数,外面的匿名函数将实时循环的i作为实参传给了形参num,函数参数传递是值传递,会将当前i的当前值复制给num,
return function(){ //同时匿名函数立即执行,返回访问num的闭包,所以结果正确。
return num;
}
}(i)
}
}
console.log( result[0]()); //0
在这里,解释下匿名函数,就是像上面那样:function没有名字,执行内容{}后面紧跟(),立即执行。
2.第二个方法,采用ES6的let方法,let声明的变量只在代码块中有效,所以可以达到效果。(讲真,这个地方不是特别理解。。。。。。。)
var result = new Array();
function func(){
for(let i=0;i<10;i++){
result[i] = function(){
return i;
}
}
}
console.log( result[0]()); //0
3.第三个方法,采用bind方法,因为用我自己的例子总输出10,没找到原因,所以先这样吧,下面这个是牛客网闭包那节中别人的解法。
//使用ES5的bind()方法
function makeClosures(arr, fn) {
var result = new Array();
for(var i=0;i<arr.length;i++){
result[i] = fn.bind(null,arr[i]);
}
return result;
}
var result = new Array();
function func(){
for(var i=0;i<10;i++){
result[i] = function(){
return i;
}.bind(i);
}
}
console.log( result[0]()); //10
另外,如果循环的是数组的话,可以用ES6的forEach().
参考:关于匿名函数和闭包,这篇文章让我恍然大悟浅谈匿名函数和闭包
再总结下,闭包的用途:一是可以读取函数内部的变量,二是可以让函数中的变量始终保存在内存中,因为闭包的存在依赖于外部函数,那么闭包存在的时候,外部函数也会存在,不会被垃圾回收机制回收。
当然这就造成了闭包的缺点,一是占用内存,解决方法是,在退出函数前,将不使用的局部变量删除,当然,根据垃圾回收机制,对不使用的全局变量,手动置为null释放引用,让值脱离执行环境,以便垃圾回收下次运行时回收。二是,由于闭包能够访问父函数,因此可以改变父函数内部变量的值,当把父函数当作对象使用,内部变量作为私有变量,闭包作为公用方法时,需要注意,不能随便改变内部变量的值。
应用场景:
1.应用场景一:setTimeout
原生的setTimeout有一个缺陷,你传递的第一个函数不能带参数。即
setTimeout(func(parma),1000);
这样的语句是不生效的(不过在不同浏览器中有不同的错误,总之都无法达到预期效果)
这时,我们就可以用闭包来实现这个效果了。
function func(param) {
return function() {
alert(param);
}
}
var f = func(1)
setTimeout(f, 1000);