前置知识:
- 每个变量在定义的时候都拥有一个作用域,该变量只能在作用域内被调用,作用域外无法跨作用域调用
- 每个变量都有生命周期,即从创建到销毁,一般变量的生命周期都与函数的执行同步。函数生命周期结束时会被内存回收机制进行清理
概念:一个函数执行完毕返回另一个未执行(或者说待执行)的函数,且返回的函数又引用了上一层(或之前几层)函数中的局部变量时,会导致引用的变量虽然生命周期已经结束,但由于它依旧被引用着而无法被内存回收机制所回收,这种现象就叫做闭包
例子
function aaa () {
// 这里定义一个典型闭包局部变量
var a = 111;
// 然后返回一个待执行函数
return function bbb () {
// 在这个函数里随便引用上面的局部变量
a += 1;
// 然后一个简单的闭包就这样产生了
}
}
作用:由于其是局部变量,对该变量的引用与访问只能由返回的函数进行,无法从外部对其调用,这就相当于赋予了这个返回的函数一个“专属”变量,这个函数可以将其称之为沟通“内外”的“桥梁”。且由于其是局部变量,命名空间是不与全局命名空间冲突的。由此就可以引申出闭包使用的场景:
- 需要让某个函数拥有“专属”存储仓库时
- 需要防止污染全局命名空间时
- 需要让函数不停收集多个变量之后再执行时
缺点:成也萧何,败也萧何。由于其是局部变量,除了使用返回的函数进行访问之外无人能够再访问它,它也无法被内存回收机制清理,这就导致了一个问题:内存泄漏。因此闭包不应该使用过多。
应用:函数的防抖与节流,函数的柯里化,react-thunk