27

闭包确实是一个说烂了的概念,校招社招都会被问到,今天总结一番。
先下定义,闭包是函数和该函数的词法作用域的组合。其实这个定义是比较教条的,可以直白的理解为闭包是一个函数,且这个函数使用了既没在它内部声明且不是它的参数的变量。
举个栗子,

function foo() { 
    var a = 2;
    function bar() { 
        console.log( a );
    }
    return bar; 
}
var baz = foo();
baz(); // 2

按照常理,foo函数在执行完毕之后会销毁掉其内部的变量a,但是bar函数内部保持着对a的引用,所以通过调用foo()把bar的引用赋给了baz,运行baz()依然可以打印出a。在这个栗子里,函数bar以及它对变量a的引用就构成了闭包。

闭包和作用域

对于闭包和作用域的关系,我的理解是闭包其实就是作用域的延伸
由于在JavaScript中函数内部可以使用函数外部的变量,所有有时候会不知不觉的产生闭包,假如在上面那个代码片段中,不允许函数内部使用函数外部的变量,闭包也就无从谈起了。

闭包有什么用?

模拟私有变量和私有方法

var Dog = (function(){
    var privateVal = 'dog'
    function doing(val) {
        console.log(privateVal + ' ' + val)
    }

    return {
        run: function(){
            doing('run')
        },
        bark: function(){
            doing('bark')
        }
    }
})()

Dog.run()    // dog run
Dog.bark()   //  dog bark

可以看到的是,run和bark这两个闭包分享了同一个词法作用域,且都引用了私有方法doing。这样,我们就可以只向外暴露run和bark两个公共接口而隐藏私有的变量和方法。

闭包与循环

或许这是面试中出现最多的问题...

for(var i = 1;i <= 5;i++) {
    setTimeout(function() {
        console.log(i)
    }, i*1000)
}
// 每隔一秒打印一个6,共打印5次

为什么事与愿违,而不是按照我们所想的依次的间隔1秒打印出1,2,3,4,5呢?首先,这段循环产生了5个闭包,而且最重要的是这5个闭包都处在同一个作用域中,也就是说它们引用的是同一个i,当for循环结束时,i变成了6。所以,5个匿名函数执行时会依次的去打印那同一个i,所以就打印出了5个6。
如何解决?
之前也说了让这5个闭包处于不同的作用域且让它们在各自的作用域中拥有它们各自的i即可。
可以使用自执行函数来创建一个新的作用域

for(var i = 1;i <= 5;i++) {
    (function(k){
        setTimeout(function() {
        console.log(k)
    }, k*1000)
    })(i)
}

在这个代码片段中,每一个setTimeout都处于一个独立的作用域中,且都引用了它们各自的k,并不是指向了外层作用域的i,所以就会打印出1,2,3,4,5
也可以使用let

for(let i = 1;i <= 5;i++) {
    setTimeout(function() {
        console.log(i)
    }, i*1000)
}

for 循环头部的 let 不仅将 i 绑定到了 for 循环的块中,事实上它将其重新绑定到了循环的每一个迭代中,确保使用上一个循环迭代结束时的值重新进行赋值。
其实使用let的本质是

for(let i = 1;i <= 5;i++) {
    let i = 上次迭代结束的i
    setTimeout(function() {
        console.log(i)
    }, i*1000)
}

其实闭包就这么多东西,而且主要是作用域的概念,作用域明白了,闭包也就明白了。
that's all, thank you.

参考资料
深入理解JavaScript系列-闭包
MDN-闭包
《你不知道的JavaScript-上卷》
「每日一题」JS 中的闭包是什么?

欢迎关注我的微信公众号-前端亚古兽

fe-yagushou.jpg


耳东
766 声望51 粉丝

知乎专栏:[链接]