什么是闭包?
- 当内部函数,在定义他的作用域外被引用时就创建了该内部函数的闭包,如果内部函数引用了外部函数的变量,当外部函数调用完毕后,这些变量在内存中不会被释放,因为闭包需要他们。
- 闭包就是能够读取其他函数内部变量的函数,或者子函数在外调用,子函数所在的父函数的作用域不会被释放。
- 闭包是嵌套的内部函数,包含被引用变量(函数)的对象
通俗一点就是如果一个函数用到了外部的变量,那么这个函数加这个变量就叫做闭包。
闭包的用途
因为作用域链的存在,函数内部可以直接读取全局变量(在顶级作用域声明的变量就是全局变量,window的属性的全局变量)。而函数内部无法读取函数内部的局部变量(在局部作用域下声明的变量就是局部变量)。当想读取函数内部的局部变量时可以使用闭包实现。
在本质上,闭包是将函数内部和函数外部连接起来的桥梁。
例如
function init() {
var name = "joney"; // name 是一个被 init 创建的局部变量
function displayName() { // displayName() 是内部函数,一个闭包
console.log((name)); // 使用了父函数中声明的变量
}
displayName();
}
init();
init()
创建了一个局部变量 name 和一个名为 displayName() 的函数。displayName() 是定义在 init() 里的内部函数,并且仅在 init() 函数体内可用。这时displayName() 就可以访问到外部函数的变量,所以 displayName() 可以使用父函数 init() 中声明的变量 name。
但闭包的用途可不止是读取函数内部的变量,它可以将函数变量保存在内存中,不会被垃圾回收机制回收。
在下面这个例子中, f1 是 f2 的父函数,而 f2 被赋给了一个全局变量,所以f2 始终在内存中,且 f2 的存在依赖于 f1中的n,这就导致变量n无法被销毁,而变量q是每次被调用时新创建的,所以每次f2执行完后它就把属于自己的变量连同自己一起销毁,于是乎最后就剩下孤零零的n。
function f1(){
var n=2;
function f2(){
var q=0;
console.log('n=',++n);
console.log('q=',++q);
}
return f2;
}
var f=f1();
f() // 3 1
f() // 4 1
闭包的优缺点
通过简单的两个例子我们可以总结一下闭包的优点和缺点:
- 优点 :保证函数不受外界干扰,实现封装,避免命名冲突;可以在内存中保存函数变量,充当缓存
- 缺点:内存消耗很大,容易造成内存泄漏, 要谨慎使用