1.闭包
- 闭包是我们函数的一种高级使用方式
- 在聊闭包之前要先回顾一下 函数
2.函数有两个阶段
1.定义阶段
2.调用阶段
3.函数的定义阶段
1.开辟一个存储空间
2.把函数体内的代码一摸一样的放在这个空间内(不解析变量)
3.把存储空间的地址给函数名
4.函数的调用阶段
1.按照函数的地址找到函数的存储空间
2.形参赋值
3.预解析
4.将函数存储空间中的代码拿出来执行(才解析变量)
5.重新定义函数的调用阶段
1.按照函数名的地址找到函数的存储空间
2.形参赋值
3.预解析
4.在内存中开辟一个执行空间
5.将函数存储空间中的代码拿出来在刚刚开辟的执行空间中执行
6.执行完毕后,内存中开辟的执行空间销毁
function fn() {
console.log('我是 fn 函数')
}
fn()
- 函数执行的时候会开辟一个 执行空间 (我们暂且把执行空间叫
xxff00
) console.log('我是 fn 函数')
这个代码就是在xxff00
这个空间中执行- 代码执行完毕以后,这个
xxff00
空间就自动销毁了
6.函数执行空间
- 每一个函数会有一个 存储空间
- 但是每一次调用都会生成一个完全不一样的 执行空间
- 并且 执行空间 会在函数执行完毕后就销毁了,但是 存储空间 不会
- 那么这个函数空间执行完毕就销毁了,还有什么意义呢?
- 我们可以有一些办法让这个空间 不销毁
- 闭包,就是要利用这个 不销毁的执行空间
7.函数执行空间不销毁
-
函数的 执行空间 会在函数执行完毕之后销毁
-
但是,一旦函数内部返回了一个 引用数据类型,并且 在函数外部有变量接受 的情况下
-
那么这个函数 执行空间 就不会销毁了
function fn() {
const obj = {
name: 'Jack',
age: 18,
gender: '男'
}
return obj
}
const o = fn()
- 函数执行的时候,会生成一个函数 执行空间 (我们暂且叫他
xxff00
) - 代码在
xxff00
空间中执行 - 在
xxff00
这个空间中声名了一个 对象空间(xxff11
) - 在
xxff00
这个执行空间把xxff11
这个对象地址返回了 - 函数外部
0
接受的是一个对象的地址没错- 但是是一个在
xxff00
函数执行空间中的xxff11
对象地址 - 因为
o
变量一直在和这个对象地址关联着,所以xxff00
这个空间一直不会销毁
- 但是是一个在
- 等到什么时候,执行一句代码
o = null
- 此时,
o
变量比在关联在xxff00
函数执行空间中的xxff11
对象地址 - 那么,这个时候函数执行空间
xxff00
就销毁了
- 此时,
8.闭包
- 闭包就是利用了这个函数执行空间不销毁的逻辑
- 有几个条件组成闭包
9.不销毁的空间
-
闭包的第一个条件就是利用了不销毁空间的逻辑
-
只不过不是返回一个 对象数据类型
-
而是返回一个 函数数据类型
function fn() { return function () {} } const f = fn()
f
变量接受的就是一个 fn的执行空间 中的 函数
10.内部函数引用外部函数中的变量
-
涉及到两个函数
-
内部函数要查看或者使用着外部函数的变量
-
function fn() { const num = 100 // 这个函数给一个名字,方便写笔记 return function a() { console.log(num) } } const f = fn()
fn()
的时候会生成一个xxff00
的执行空间- 在
xxff00
这个执行空间内部,定义了一个a
函数的 存储空间xxff11
- 全局 f 变量接受的就是
xxff00
里面的xxff11
- 所以
xxff00
就是不会销毁的空间 - 因为
xxff00
不会销毁,所以,定义再里面的变量 num 也不会销毁 - 将来
f()
的时候,就能访问到 num 变量
11.闭包的特点
- 为什么要叫做特点,就是因为他的每一个点都是优点同时也是缺点
- 作用域空间不销毁
- 优点: 因为不销毁,变量页不会销毁,增加了变量的生命周期
- 缺点: 因为不销毁,会一直占用内存,多了以后就会导致内存溢出
- 可以利用闭包访问再一个函数外部访问函数内部的变量
- 优点: 可以再函数外部访问内部数据
- 缺点: 必须要时刻保持引用,导致函数执行栈不被销毁
- 保护私有变量
- 优点: 可以把一些变量放在函数里面,不会污染全局
- 缺点: 要利用闭包函数才能访问,不是很方便
- 作用域空间不销毁
闭包概念(熟读并背诵全文)
- 有一个 A 函数,再 A 函数内部返回一个 B 函数
- 在A 函数外部有变量引用这个 B 函数
- B 函数内部访问着 A 函数内部的私有变量
- 以上三个条件缺一不可