实现Promise的详细流程(实现一个最完整的Promise构造函数)

一、创建promise构造函数基本架构

  • 获取状态

    status = 'pending' 初始化 'resolved' 已获取

    'rejected' 获取失败

  • 初始数据

    data = undefined

  • 定义成功/失败回调函数容器

    onResolved = undefined onRejected = undefined

  • 定义成功/失败函数 resolve/reject 内部

    _this.status更改状态为已获取/获取失败 _this.data保存成功/失败数据

    • 如果成功/失败回调函数已定义 立即异步执行成功/失败回调
  • 执行器执行成功/失败函数 resolve/reject

    • 如捕获到异常则代表then retrun promise的结果为失败

      执行reject, 传入错误

二、定义构造函数原型then函数, 接收成功/失败回调 onResolved/onRejected, 最终then返回一个新的promise实例

  • 定义return promise结构, 接收成功/失败函数参数, return promise的执行结果由onResolved, onRejected决定

    1. return promise定义DealWithThenReturnPromise(return promise状态处理函数)并接受then的成功/失败回调 ====>onResolved, onRejected 有三种结果情况来改变return pormise的状态

      执行结果是异常抛出, 执行return promise的失败函数, 并传入异常数据

      执行结果是promise实例, 执行return promise是否调用成功/失败函数, 由执行结果的promise的结果决定

      执行结果不是promise实例, 执行return promise的成功函数, 并传入执行结果

      执行then传入的成功/失败回调, 并将返回值保存为result

      1. result为Promise的实例

        执行result实例then方法, 并在其成功/失败回调传入return promise的成功/失败函数

      2. result不是promise实例 执行return promise成功函数,传入result
      3. result执行结果是异常抛出 捕获异常并执行return promise失败函数, 传入error
  • return promise定义三种数据获取状态处理方式

    ====>数据获取有三种情况会发生, 并且每一种状态都得是异步执行 数据已经获取成功, 代表可以直接执行并获取到onResolved返回值 数据获取失败, 代表可以直接执行并获取到onRejected返回值 数据并没有获取, 代表onRejected,onResolved并不能马上执行, 要存入实例当中由执行器的回调函数执行

    1. 数据获取成功 执行return promise状态处理函数, 并传入成功回调函数
    2. 数据获取失败 执行return promise状态处理函数, 并传入失败回调函数
    3. 数据暂未获取 封装成功/失败函数, 在封装函数内执行return promise状态处理函数, 并传入then成功/失败回调 这个封装函数最终由当前promise实例的成功/回调函数执行 并最终会由return promise状态处理函数处理数据

三、定义Promise.resolve

Promise.resolve接收三类参数, 并返回一个新的promise实例 成功的Promise实例 失败的Promise实例 不是Promise实例的任意值

Promise.resolve return promise架构 判断传参是不是Promise实例, 如果是, 调用then方法, 并绑定return promise的成功/失败函数 参数不是Promise, 直接调用return Promise成功函数, 并传入参数

四、定义Promise.all

Promise.all接收一个Promise或其他值数组 如果全部成功, 则执行return promise并传入Promise成功的结果, 只要有一个失败, 则执行失败的return Promise 传入失败值

Promise.all return promise架构 定义一个数组容器 定义一个计数器 遍历数组(forEach), 每次进入循环计数器+1 执行每个promise实例then,如果成功,在将值根据下标存入数组容器 判断promise数组length是否等于计数器,如果等于,代表需要执行return promise的成功函数 如果then执行的是失败,则直接调用return promise 失败函数,并传入then失败值

五、定义promise.rule

Promise.rule接收一个Promise或其他值数组 如果一个成功/失败,则直接调用return promise 成功/失败函数

实现具体代码

//! 定义常用常量
//? 未获取, 代表当前promise的resolve是异步完成的
const PENDING = 'pending'
//? 已获取, 代表当前promise是同步执行的
const RESOLVED = 'resolved'
//? 失败, 代表当前promise的执行结果为失败
const REJECTED = 'rejected'
//! 三种状态只能从pending中更改一次

function Promise(executor) {
    const _this = this
    //! Promise构造函数定义初始化状态为 未获取状态
    _this.status = PENDING
    //! Promise构造函数定义初始化数据为undefined
    _this.data = undefined
    //! Promise构造函数定义初始化成功容器为undefined
    _this.onResolved = undefined
    //! Promise构造函数定义初始化失败容器为undefined
    _this.onRejected = undefined
    //! 获取成功函数
    function resolve(value) {
        //! 如果状态不是pending, 则直接返回
        if (_this.status !== PENDING) { return }
        //! 将状态更改为获取成功
        _this.status = RESOLVED
        //! 将数据保存
        _this.data = value
        //! 如果成功函数已经定义, 代表需要异步执行成功回调 onResolved 并传入数据
        if (_this.onResolved) {
            setTimeout(() => _this.onResolved(value));
        }
    }
    //! 获取失败函数
    function reject(reason) {
        //! 如果状态不是pending, 则直接返回
        if (_this.status !== PENDING) { return }
        //! 将状态更改为获取失败
        _this.status = REJECTED
        //! 将数据保存
        _this.data = reason
        //! 如果失败函数已经定义, 代表需要异步执行失败回调 onResolved 并传入数据
        if (_this.onRejected) {
            setTimeout(() => _this.onRejected(reason));
        }
    }
    try {
        executor(resolve, reject)
    } catch (error) {
        //! 如果捕获到错误 则代表then retrun promise的结果为失败
        //! 那么直接调用retrun promise的失败函数, 向下传递失败
        reject(error)
    }
}
Promise.prototype = {
    then: function (onResolved, onRejected) {
        //! onResolved 成功回调; onRejected 失败回调;
        const _this = this
        //! 当执行then, 返回一个新的promise
        //! return promise的执行结果由onResolved, onRejected决定
        return new Promise((resolve, reject) => {
            /* 
            !   ----->数据获取有三种情况会发生
            !   数据已经获取成功, 代表可以直接执行并获取到onResolved返回值
            !   数据获取失败, 代表可以直接执行并获取到onRejected返回值
            !   数据并没有获取, 代表onRejected,onResolved并不能马上执行, 要存入实例当中由执行器的回调函数执行
            !   ----->并且每一种状态都得是异步执行
            */
            if (_this.status === RESOLVED) {
                //! 数据获取成功, 执行return promise状态处理函数, 并传入成功回调
                setTimeout(() => DealWithThenReturnPromise(onResolved));
            } else if (this.status === REJECTED) {
                //! 数据获取失败, 执行return promise状态处理函数, 并传入失败回调
                setTimeout(() => DealWithThenReturnPromise(onRejected));
            } else {
                //! 数据暂未获取, 封装成功/失败函数, 在封装函数内执行return promise状态处理函数, 并传入then成功/失败回调,
                //! 这个封装函数最终由当前promise实例的成功/回调函数执行
                //! 当获取数据的时候, 会在DealWithThenReturnPromise函数中获取
                _this.onResolved = value => DealWithThenReturnPromise(onResolved)
                _this.onRejected = reason => DealWithThenReturnPromise(onRejected)
            }
            //! 定义处理return promise 函数, 传入要处理的成功|失败函数
            function DealWithThenReturnPromise(callback) {
                /* 
                !   onResolved, onRejected 有三种结果情况来改变return pormise的状态
                !   执行结果是异常抛出, 执行return promise的失败函数, 并传入异常数据
                !   执行结果是promise实例, 执行return promise是否调用成功/失败函数, 由执行结果的promise的结果决定
                !   执行结果不是promise实例, 执行return promise的成功函数, 并传入执行结果
                */
                try {
                    //! 将then成功/失败返回值保存
                    const result = callback(_this.data)
                    if (result instanceof Promise) {
                        //! 执行结果是promise实例, 由执行结果promise实例决定return promise的执行结果
                        //! 易懂写法 result.then(value=> resolve(value), reason=> reject(reason))

                        //! 简便写法
                        result.then(resolve, reject)
                        //! 这里将return promise的成功/失败回调传入执行结果实例then中, 所以这里的resolve有双层意义, 
                        //! 一是结果实例then的onResolved与onRejected, 
                        //! 二是return promise的resolve与reject
                    } else {
                        //! 执行结果不是promise实例, 执行return promise成功函数, 传入执行结果
                        resolve(result)
                    }
                } catch (error) {
                    //! 执行结果是异常抛出, 捕获异常并执行 return promise 失败函数, 传入异常
                    reject(error)
                }
            }

        })
    },
    catch: function (reject) { return this.then(null, reject) },
}
Promise.resolve = function (value) {
    /* 
    ! Promise.resolve接收有三种参数
    ! 1.实例结果是失败的
    ! 2.实例结果是成功的
    ! 3.不是实例
    */
    return new Promise((resolve, reject) => {
        if (value instanceof Promise) {
            //! 如果value是Promise的实例
            value.then(resolve, reject)
        } else {
            //! 如果不是
            resolve(value)
        }
    })

}
Promise.reject = function (reason) {
    return new Promise((resolve, reject) => {
        reject(reason)
    })
}
Promise.all = function (promises) {
    //! 创建一个计数器
    let resolveCount = 0
    //! 创建数组指定长度
    const values = new Array(promises.length)
    return new Promise((resolve, reject) => {
        promises.forEach((item, index) => {
            Promise.resolve(item).then(
                value => {
                    //! 当获取成功时, 计数器+1
                    resolveCount++
                    //! promise实例成功, 将结果保存在数组
                    values[index] = value
                    //! 当执行到最后一个, 执行all return promise成功函数, 传入成功值的数组
                    promises.length === resolveCount ? resolve(values) : []
                },
                //! 只要有一个失败, 那么就all返回的promise就是失败
                reason => reject(reason)
            )
        })
    })
}
Promise.rule = function (promises) {
    return new Promise((resolve, reject) => {
        promises.forEach(item => {
            Promise.resolve(item).then(resolve, reject)
        })
    })
}

你可能感兴趣的:(promise,javascript,前端,node.js)