实现一个简易版 Promise
在完成符合Promise/A+ 规范的代码之前,我们可以先来实现一个简易版 Promise
,因为在面试中,如果你能实现出一个简易版的 Promise
基本可以过关了。
那么我们先来搭建构建函数的大体框架
1 | const PENDING = 'pending' |
- 首先我们创建了三个常量用于表示状态,对于经常使用的一些值都应该通过常量来管理,便于开发及后期维护
- 在函数体内部首先创建了常量
that
,因为代码可能会异步执行,用于获取正确的this
对象 - 一开始
Promise
的状态应该是pending
value
变量用于保存resolve
或者reject
中传入的值resolvedCallbacks
和rejectCallbacks
用于保存then
中的回调,因为当执行完Promise
时状态可能还是等待中,这时候应该把then
中的回调保存起来用于状态改变时使用
接下来我们来完善 resolve
和 reject
函数,添加在 MyPromise
函数体内部
1 | function resolve(value) { |
- 首先两个函数都得判断当前状态是否为等待中,因为规范规定只有等待态才可以改变状态
- 将当前状态更改为对应状态,并且将传入的值赋值给
value
- 遍历回调函数组并执行
完成以上两个函数以后,我们就该实现如何执行 Promise
中传入的函数了
1 | try { |
- 实现很简单,执行传入的参数并且将之前两个函数当做参数穿进去
- 要注意的是,可能执行函数过程中会遇到错误,需要捕获错误并且执行
reject
函数
最后我们来实现较为复杂的 then
函数
1 | MyPromise.prototype.then = function(onFulfilled, onRejected) { |
首先判断两个参数是否为函数类型,因为这两个参数是可选参数
当参数不是函数类型时,需要创建一个函数赋值给对应的参数,同时也实现了透传,比如如下代码
1
2
3// 该代码目前在简单版中会报错
// 只是作为一个透传的例子
Promise.resove(4).then().then(value => console.log(value))接下来就是一系列判断状态的逻辑,当状态不是等待态时,就去执行相对应的函数。如果状态是等待态的话,就往回调函数中
push
函数,比如如下代码就会进入等待态的逻辑1
2
3
4
5
6
7new MyPromise((resolve, reject) => {
setTimeout(() => {
resolve(1)
}, 0)
}).then(value => {
console.log(value)
})以上就是简单版
Promise
实现,接下来一小节是实现完整版Promise
的解析,相信看完完整版的你,一定会对于Promise
的理解更上一层楼
简易版综合代码:
1 | // 先定义三个常量 |
实现一个符合 Promise/A+ 规范的 Promise
这小节代码需要大家配合规范阅读,因为大部分代码都是根据规范去实现的。
我们先来改造一下 resolve
和 reject
函数
1 | function resolve(value) { |
- 对于
resolve
函数来说,首先需要判断传入的值是否为Promise
类型 - 为了保证函数执行顺序,需要将两个函数体代码使用
setTimeout
包裹起来
接下来继续改造 then
函数中的代码,首先我们需要新增一个变量 promise2
,因为每个 then
函数都需要返回一个新的 Promise
对象,该变量用于保存新的返回对象,然后我们先来改造判断等待态的逻辑
1 | if (that.state === PENDING) { |
- 首先我们返回了一个新的
Promise
对象,并在Promise
中传入了一个函数 - 函数的基本逻辑还是和之前一样,往回调数组中
push
函数 - 同样,在执行函数的过程中可能会遇到错误,所以使用了
try...catch
包裹 - 规范规定,执行
onFulfilled
或者onRejected
函数时会返回一个x
,并且执行Promise
解决过程,这是为了不同的Promise
都可以兼容使用,比如 JQuery 的Promise
能兼容 ES6 的Promise
接下来我们改造判断执行态的逻辑
1 | if (that.state === RESOLVED) { |
- 其实大家可以发现这段代码和判断等待态的逻辑基本一致,无非是传入的函数的函数体需要异步执行,这也是规范规定的
- 对于判断拒绝态的逻辑这里就不一一赘述了,留给大家自己完成这个作业
最后,当然也是最难的一部分,也就是实现兼容多种 Promise
的 resolutionProcedure
函数
1 | function resolutionProcedure(promise2, x, resolve, reject) { |
首先规范规定了 x
不能与 promise2
相等,这样会发生循环引用的问题,比如如下代码
1 | let p = new MyPromise((resolve, reject) => { |
然后需要判断 x
的类型
1 | if (x instanceof MyPromise) { |
这里的代码是完全按照规范实现的。如果 x
为 Promise
的话,需要判断以下几个情况:
- 如果
x
处于等待态,Promise
需保持为等待态直至x
被执行或拒绝 - 如果
x
处于其他状态,则用相同的值处理Promise
当然以上这些是规范需要我们判断的情况,实际上我们不判断状态也是可行的。
接下来我们继续按照规范来实现剩余的代码
1 | let called = false |
- 首先创建一个变量
called
用于判断是否已经调用过函数 - 然后判断
x
是否为对象或者函数,如果都不是的话,将x
传入resolve
中 - 如果
x
是对象或者函数的话,先把x.then
赋值给then
,然后判断then
的类型,如果不是函数类型的话,就将x
传入resolve
中 - 如果
then
是函数类型的话,就将x
作为函数的作用域this
调用之,并且传递两个回调函数作为参数,第一个参数叫做resolvePromise
,第二个参数叫做rejectPromise
,两个回调函数都需要判断是否已经执行过函数,然后进行相应的逻辑 - 以上代码在执行的过程中如果抛错了,将错误传入
reject
函数中