闭包概念
- 有一个 A 函数,再 A 函数内部返回一个 B 函数
- 再 A 函数外部有变量引用这个 B 函数
- B 函数内部访问着 A 函数内部的私有变量
函数作用域(闭包前置知识)
在聊闭包之前先回顾一下函数,函数有两个阶段:a.定义阶段,b.调用阶段。要学会闭包函数一定要弄懂。
函数定义阶段
- 开辟一个 存储空间
- 把函数体内的代码一模一样的放在这个空间内(不解析变量)
- 把 存储空间 的地址给函数名
函数调用阶段
-
按照函数名的地址找到函数的 存储空间
-
形参赋值
-
预解析
-
在内存中开辟一个 执行空间
-
将函数 存储空间 中的代码拿出来在刚刚开辟的 执行空间 中执行
-
执行完毕后,内存中开辟的 执行空间 销毁
function fn() {
console.log('我是 fn 函数')
}
fn()
- 函数执行的时候会开辟一个 执行空间 (暂且叫他
xxff00
) console.log('我是 fn 函数')
这个代码就是在xxff00
这个空间中执行- 代码执行完毕以后,这个
xxff00
空间就销毁了
函数执行空间
- 每一个函数会有一个 存储空间
- 但是每一次调用都会生成一个完全不一样的 执行空间
- 并且 执行空间 会在函数执行完毕后就销毁了,但是 存储空间 不会
- 那么这个函数空间执行完毕就销毁了,还有什么意义呢?
- 我们可以有一些办法让这个空间 不销毁
- 闭包,就是要利用这个 不销毁的执行空间
函数执行空间不销毁
-
函数的 执行空间 会在函数执行完毕之后销毁
-
但是,一旦函数内部返回了一个 引用数据类型,并且 在函数外部有变量接受 的情况下
-
那么这个函数 执行空间 就不会销毁了
function fn() {
var num = 100
console.log('你好 世界')
// 返回了一个复杂数据类型
return [1, 2, 3]
}
// res 接受的就是 函数fn 返回的那个复杂数据类型(一个数组)
// => res 接受的是 函数fn 执行空间里面那个数组的地址xxff11
var res = fn()
/*
res 接受的是 函数fn 的执行空间内的一个数组
+ 只要这个数组还存在, 就表示这个函数执行空间没有销毁
+ 因为一旦函数执行空间销毁了, 那么这个数组也就没有了
+ 所以这个函数执行空间不能销毁
*/
res = null
/*
当 res 指向别的位置了, 不在是 xxff11 空间里面的那个数组了
+ 也就是函数执行空间内返回的复杂数据类型, 外面没有接受的
+ 那么这个函数执行空间就销毁了
*/
闭包
不销毁的空间
-
闭包的第一个条件就是利用了不销毁空间的逻辑
-
只不过不是返回一个 对象数据类型
-
而是返回一个 函数数据类型
function a() {
// 这个 num 变量就是函数 a 的私有变量
var num = 100
return function b() {
console.log(num)
}
}
// res 接受的是 a 函数执行以后的返回值
// res 接受的就是函数 a 内部返回的一个复杂数据类型(函数b)
// 导致函数 a 的执行空间不会销毁
var res = a()
// 从现在开始, res 随时可以是一个 函数a 里面返回的 函数b
// res 随时可以调用
res()
// 当 res 调用的时候, 打印 num
// 打印出来的就是 a 函数内部的私有变量 num 的值
如何产生闭包?
- 在嵌套内部函数定于并引用父函数的属性时,就产生了闭包。
- 产生闭包必须要有嵌套函数,以及子函数引用父函数的属性或者变量才能产生闭包。
闭包的特点
- 为什么要叫做特点,就是因为他的每一个点都是优点同时也是缺点
- 延长了变量的生命周期
- 优点: 因为执行空间不销毁, 变量也没有销毁
- 缺点: 因为执行空间不销毁, 会一直存在在内存中
- 可以访问函数内部的私有变量
- 优点: 利用闭包函数可以访问函数内部的私有变量
- 缺点: 执行空间不会销毁, 会一直存在在内存中
- 保护私有变量(只要是函数, 就有这个特点)
- 优点: 保护私有变量不被外界访问
- 缺点: 如果向访问, 必须要利用闭包函数
闭包的函数的缺点 致命
- 因为当一段内存空间中有一个不会被销毁的东西一直存在
- 那么就会出现内存占用, 如果过多, 就会导致内存溢出
- 那么结果就是 内存泄漏
闭包的作用 - 就是当你需要延长变量的声明周期的时候
- 或者你需要访问某一个函数内部的私有数据的时候
- 你可以使用 闭包 函数来解决
关键字:闭包 函数