是什么
一个函数和对其周围状态(词法环境)的引用捆绑在一起(函数被引用包围),这样的组合我们称之为闭包
也就是说,闭包让你可以在一个内层函数中访问到其外层函数的作用域
🍖在 JavaScript中,每当创建一个函数,闭包就会在函数创建的同时被创建出来,作为函数内部与外部连接起来的一座桥梁
示例:
function init(){
var name = "Mozilla";
function displayName(){
alert(name);
}
displayName();
}
init();
任何闭包的使用场景都离不开这两点:
- 创建私有变量
- 延长变量的生命周期
一般函数的词法环境在函数返回后就被销毁,但是闭包会保存对创建时所在词法环境的引用,即便创建时所在的执行上下文被销毁,但创建时所在词法环境依然存在,以达到延长变量的生命周期的目的。
例子:
function makeSizer(size){
return function(){
document.body.style.fontSize = size + 'px'
}
}
var size12 = makeSizer(12);
document.getElementById("size-12").onclick = size12;
柯里化函数
柯里化的目的在于避免频繁调用具有相同参数函数的同时,进行重用。
// 假设求长方形面积
function getArea(width,height){
return width * height
}
const area1 = getArea(10,20)
// 如果宽老是10
const area2 = getArea(10,30)
//我们可以使用闭包柯里化这个计算面积的函数
function getArea(width){
return height=>{
return width * height
}
}
const getTenWidthArea = getArea(10)
// 只要是宽度为10的长方形就可以这样计算
const area1 = getTenWidthArea(20)
使用闭包模拟私有方法
🍚🍚在JavaScript中,没有支持声明私有变量,但我们可以使用闭包来模拟私有方法
其他
例如计数器、延迟调用、回调等闭包的应用,其核心思想还是创建私有变量和延长变量的生命周期
注意事项
如果不是某些特定任务需要使用闭包,在其它函数中创建函数是不明智的,因为闭包在处理速度和内存消耗方面对脚本性能具有负面影响
例如,在创建新的对象或者类时,方法通常应该关联于对象的原型,而不是定义到对象的构造器中
原因在于每个对象的创建,方法都会被重新赋值
上面的代码中,我们并没有利用到闭包的好处,因此可以避免使用闭包。修改成如下