闭包的定义
在了解闭包之前,我们首先要搞懂JavaScript的变量作用域的一些问题。
变量作用域无非就是两种:
- 全局作用域(全局变量)
- 局部作用域(局部变量)
用于 JavaScript语言 特殊之处在于:函数内部可以直接访问全局变量(var / const 定义的变量);另一方面,函数内部无法让外部使用。
所以,当我们外部需要使用函数内部定义的变量时,就需要用到“闭包”
在函数内部再定义一个函数
function f1(){
var n = 99;
function f2(){
console.log(n);
}
return f2;
}
var ff = f1()
ff() // 输出的为 99
在函数中 f1 内部所有变量对 f2 都是可见的。在 f2 内部的变量 对 f1 都是 不可见的。这是由于JavaScript语言具有 “链式作用域”结构。子对象会一级级的寻找父对象,父级对象无法访问子对象中的内容。
以上就可以理解为 闭包。
其实可以简单的理解为:一个能够读取其他函数内部变量的函数。
由于JavaScript 语言中,只有函数内部子函数才能读取局部变量,因此可以理解为:定义在函数内部的函数。
本质上来讲,闭包就是将函数内部和函数外部连接起来的桥梁
闭包的用途
1、读取函数内部的变量;
2、让变量值始终保持在内存中,避免被污染。
function f1(){
var n = 99;
nAdd = function(){n = n+1}
function f2(){
console.log(n);
}
return f2;
}
var ff = f1()
ff() // 99
nAdd()
ff() // 100
f1() 是 f2() 的父函数,f2被赋予的是全局变量,这就导致 f2始终在内存中,而f2的存在又依赖于 f1 ,故而 f1也始终在内存之中存在,不会随着函数调用结束而被垃圾回收机制回收掉。并且 nAdd 也是一个全局变量。