手写Promise源码

Promise类核心逻辑实现

/*
    1. Promise就是一个类,在执行这个类的时候,需要传递一个执行器进去,执行器会立即执行
    2. Promise中有三种状态:成功fulfilled、失败rejected、等待pending
        pending -> fulfilled
        pending -> rejected
        一旦状态确定就不可更改
    3. resolve和reject函数是用来更改状态的
        resolve:fulfilled
        reject:rejected
    4. then方法内部做的事情就是判断状态,如果状态是成功就调用成功回调函数,如果状态是失败就调用失败回调函数。then方法是被定义在原型对象中的
    5. then成功回调有一个参数,表示成功之后的值;失败回调有一个参数,表示失败后的原因
*/

const MyPromise = require('./MyPromise')

let promise = new MyPromise((resolve, reject) => {
    resolve('成功')
    reject('失败')
})

promise.then(value => {
    console.log(value)
}, reason => {
    console.log(reason)
})
// MyPromise.js
const PENDING = 'pending'  // 等待
const FULFILLED = 'fulfilled'  // 成功
const REJECTED = 'rejected'  // 失败


class MyPromise {
    constructor(executor) {
        // 执行器立即执行,因此在此处调用传入的执行器函数
        executor(this.resolve, this.reject)
    }
    // Promise状态
    status = PENDING
    // 成功之后的值
    value = undefined
    // 失败之后的原因
    reason = undefined

    resolve = value => {
        // 如果状态不是等待,阻止程序向下执行
        if (this.status !== PENDING) return
        // 将状态更改为成功
        this.status = FULFILLED
        // 保存成功之后的值
        this.value = value
    }

    reject = reason => {
        // 如果状态不是等待,阻止程序向下执行
        if (this.status !== PENDING) return
        // 将状态更改为失败
        this.status = REJECTED
        // 保存失败后的原因
        this.reason = reason
    }

    then(successCallback, failCallback) {
        // 判断状态
        if (this.status === FULFILLED) {
            successCallback(this.value)
        } else if (this.status === REJECTED) {
            failCallback(this.reason)
        }
    }
}

module.exports = MyPromise

在Promise类中加入异步逻辑

如果有异步情况,那么需要在.then()方法中加入对PENDING状态的判断,由于resolve/reject还未执行,需要先将回调函数存储起来,等到resolve/reject执行的时候调用。

// 如果执行器函数中是异步
let promise = new MyPromise((resolve, reject) => {
    setTimeout(() => {
        resolve('成功')
    }, 1000);
})
class MyPromise {
    // ...
    // 成功回调函数
    successCallback = undefined
    // 失败回调函数
    failCallback = undefined

    resolve = value => {
        // ...
        // 判断成功回调是否存在,如果存在则调用
        this.successCallback && this.successCallback(this.value)
    }

    reject = reason => {
        // ...
        // 判断失败回调是否存在,如果存在则调用
        this.failCallback && this.failCallback(this.reason)
    }

    then(successCallback, failCallback) {
        // 判断状态
        if (this.status === FULFILLED) {
            successCallback(this.value)
        } else if (this.status === REJECTED) {
            failCallback(this.reason)
        } else {
            // 等待
            // 将成功回调和失败回调存储起来
            this.successCallback = successCallback
            this.failCallback = failCallback
        }
    }
}

实现then方法多次调用添加多个处理函数

如果.then()方法被多次调用,那么需要存储多个处理函数,因此将MyPromise类中的successCallbackfailCallback改成数组。

// 如果then被多次调用
promise.then(value => {
    console.log(value)
}, reason => {
    console.log(reason)
})

promise.then(value => {
    console.log(value)
}, reason => {
    console.log(reason)
})

promise.then(value => {
    console.log(value)
}, reason => {
    console.log(reason)
})
class MyPromise {
    // ...
    // 成功回调函数
    successCallback = []
    // 失败回调函数
    failCallback = []

    resolve = value => {
        // ...
        // 判断成功回调是否存在,如果存在则调用
        // this.successCallback && this.successCallback(this.value)
        while (this.successCallback.length !== 0) this.successCallback.shift()(this.value)
    }

    reject = reason => {
        // ...
        // 判断失败回调是否存在,如果存在则调用
        // this.failCallback && this.failCallback(this.reason)
        while (this.failCallback.length !== 0) this.failCallback.shift()(this.reason)
    }

    then(successCallback, failCallback) {
        // 判断状态
        if (this.status === FULFILLED) {
            successCallback(this.value)
        } else if (this.status === REJECTED) {
            failCallback(this.reason)
        } else {
            // 等待
            // 将成功回调和失败回调存储起来
            this.successCallback.push(successCallback)
            this.failCallback.push(failCallback)
        }
    }
}

module.exports = MyPromise

实现then方法的链式调用

要实现.then()方法的链式调用,那么它返回的必须是Promise对象,因此需要改写then方法,创建一个新的Promise对象,然后通过resolve将本次结果传递给下次调用。

// then方法的链式调用
function other() {
    return new MyPromise((resolve, reject) => {
        resolve('other')
    })
}
promise.then(value => {
    console.log(value)
    return other()
}).then(value => {
    console.log(value)
})
class MyPromise {
    // ...
    then(successCallback, failCallback) {
        let promise2 = new MyPromise((resolve, reject) => {
            // 判断状态
            if (this.status === FULFILLED) {
                let x = successCallback(this.value)
                // 判断x的值是普通值还是Promise对象
                // 如果是普通值,直接调用resolve
                // 如果是Promise对象,查看Promise对象返回的结果
                // 再根据Promise对象返回的结果决定调用resolve还是reject
                resolvePromise(x, resolve, reject)
            } 
            // ...
        })
        return promise2
    }
}

function resolvePromise(x, resolve, reject) {
    if (x instanceof MyPromise) {
        // Promise对象
        // x.then(value => resolve(value), reason => reject(reason))
        x.then(resolve, reject)
    } else {
        // 普通值
        resolve(x)
    }
}

then方法链式调用识别Promise对象自返回

let p1 = promise.then(value => {
    console.log(value)
    return p1
})

p1.then(value => {
    console.log(value)
}, reason => {
    console.log(reason)
})
class MyPromise {
    // ...
    then(successCallback, failCallback) {
        let promise2 = new MyPromise((resolve, reject) => {
            // 判断状态
            if (this.status === FULFILLED) {
                // 为了能够取到promise2,将此段代码改成异步
                setTimeout(() => {
                    let x = successCallback(this.value)
                    // 判断x的值是普通值还是Promise对象
                    // 如果是普通值,直接调用resolve
                    // 如果是Promise对象,查看Promise对象返回的结果
                    // 再根据Promise对象返回的结果决定调用resolve还是reject
                    resolvePromise(promise2, x, resolve, reject)
                }, 0);
            } 
            // ...
        })
        return promise2
    }
}

function resolvePromise(promise2, x, resolve, reject) {
    // 如果当前Promise对象自返回,则报错
    if (promise2 === x) {
        return reject(new TypeError('Chaining cycle detected for promise #<Promise>'))
    }
    // ...
}

捕获错误及then链式调用其他状态代码补充

1.执行器中的错误捕获;
2.then的回调函数报错,要在下一个then方法中捕获到。

// 错误捕获
let promise = new Promise((resolve, reject) => {
    // 第一个
    throw new Error('executor error')
    resolve('成功')
})

promise.then(value => {
    // 第二个
    throw new Error('then error')
    console.log(value)
}, reason => {
    console.log(reason)
}).then(value => {
    console.log(value)
}, reason => {
    console.log('第二个回调函数捕获错误')
    console.log(reason.message);
})
class MyPromise {
    constructor(executor) {
        try {
            // 执行器立即执行,因此在此处调用传入的执行器函数
            executor(this.resolve, this.reject)
        } catch (e) {
            this.reject(e)
        }
    }
    // ...

    resolve = value => {
        // ...
        while (this.successCallback.length !== 0) this.successCallback.shift()()
    }

    reject = reason => {
        // ...
        while (this.failCallback.length !== 0) this.failCallback.shift()()
    }

    then(successCallback, failCallback) {
        let promise2 = new MyPromise((resolve, reject) => {
            // 判断状态
            if (this.status === FULFILLED) {
                // 为了能够取到promise2,将此段代码改成异步
                setTimeout(() => {
                    try {
                        let x = successCallback(this.value)
                        // 判断x的值是普通值还是Promise对象
                        // 如果是普通值,直接调用resolve
                        // 如果是Promise对象,查看Promise对象返回的结果
                        // 再根据Promise对象返回的结果决定调用resolve还是reject
                        resolvePromise(promise2, x, resolve, reject)
                    } catch (e) {
                        reject(e)
                    }
                }, 0)
            } else if (this.status === REJECTED) {
                setTimeout(() => {
                    try {
                        let x = failCallback(this.reason)
                        resolvePromise(promise2, x, resolve, reject)
                    } catch (e) {
                        reject(e)
                    }
                }, 0)
            } else {
                // 等待
                // 将成功回调和失败回调存储起来
                this.successCallback.push(() => {
                    setTimeout(() => {
                        try {
                            let x = successCallback(this.value)
                            resolvePromise(promise2, x, resolve, reject)
                        } catch (e) {
                            reject(e)
                        }
                    }, 0)
                })
                this.failCallback.push(() => {
                    setTimeout(() => {
                        try {
                            let x = failCallback(this.reason)
                            resolvePromise(promise2, x, resolve, reject)
                        } catch (e) {
                            reject(e)
                        }
                    }, 0)
                })
            }
        })
        return promise2
    }
}

将then方法的参数变成可选参数

// then方法不传参
promise.then().then().then(value => {
    console.log(value)
}, reason => {
    console.log(reason)
})
then(successCallback, failCallback) {
    // then方法可以不传参,因此需要先判断
    successCallback = successCallback ? successCallback : value => value
    failCallback = failCallback ? failCallback : reason => { throw reason }  // 使用throw抛出异常
    // ...

Promise.all()方法的实现

static all(array) {
    let result = []
    let index = 0
    return new MyPromise((resolve, reject) => {
        function addData(key, value) {
            result[key] = value
            index++
            // 防止Promise对象中有异步操作返回空值
            if (index === array.length) {
                resolve(result)
            }
        }
        for (let i = 0; i < array.length; i++) {
            let current = array[i]
            if (current instanceof MyPromise) {
                // Promise对象
                current.then(value => addData(i, value), reason => reject(reason))
            } else {
                // 普通值
                addData(i, current)
            }
        }
    })
}

Promise.resolve方法的实现

static resolve(value) {
    if (value instanceof MyPromise) return value
    return new MyPromise(resolve => resolve(value))
}

Promise.finally方法的实现

finally(callback) {
    // 不管是否成功都调用回调函数
    // 调用this.then方法得到当前Promise对象的状态
    // finally方法之后可以链式调用then,因此需要返回Promise对象
    return this.then(value => {
        // 第一种写法
        // callback()
        // return value

        // 第二种写法
        return MyPromise.resolve(callback()).then(() => value)
        // callback函数中可能有异步操作,因此需要等待callback执行完毕再返回value
    }, reason => {
        // callback()
        // throw reason
        return MyPromise.resolve(callback()).then(() => { throw reason })
    })
}

catch方法的实现

catch(failCallback) {
    return this.then(undefined, failCallback)
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,816评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,729评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,300评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,780评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,890评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,084评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,151评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,912评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,355评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,666评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,809评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,504评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,150评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,121评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,628评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,724评论 2 351

推荐阅读更多精彩内容