Vue

手写Promise(自定义Promise)

2020-07-23  本文已影响0人  Poppy11
首先我们得先理解Promise工作的机制,其实Promise就是一个构造函数

它的内部有三个状态

而想改变这三个状态必须要通过resolve()或者reject()这两个方法,resolve()可以将pending转换为resolved,rejected()可以将pending转换为rejected。并且将得到的数值存储在内部的data里面。并且这状态一旦转换是不可逆的。

Promsie的原型对象含有then,catch这两个方法

然后就可以开始自定义实现Promise了

一、大致框架列出来

(function () {
const PENDING = 'pending'
const RESOLVED = 'resolved'
const REJECTED = 'rejected'
function Promise(executor){
     this.status = PENDING
     this.data = undefind
     this.callbacks = []
      resolve = () => {
      }
      reject = () => {
      }
      try {
          executor(resolve, reject)
      } catch (e) {
          reject(e)
      }
     Promise.prototype.then = () => {
     }
     Promise.prototype.catch = () => {
     }
      /*Promise函数对象的方法*/
     Promise.resolve = function (value) {

     }
     Promise.reject = function (error) {

     }
    /*只有当所有promise成功时才成功*/
     Promise.all = function (promises) {

     }
    /*其结果由第一个完成的promise决定*/
     Promise.race = function (promises) {
     }
executor(resolve,reject)
}
window.Promise = Promise
})()

二、写函数内部resolve/reject方法

resolve = (value) => {
    if(this.satus !== PENDING){
       return
    }
    this.status = RESOLVED
    this.data = value
   if (this.callbacks.length > 0) {
                setTimeout(() => {
                    this.callbacks.forEach(callbacksObj => {
                        callbacksObj.onResolved(value)
                    })
                })
            }
}
reject = (error) => {
    if(this.satus !== PENDING){
       return
    }
    this.status = REJECTED
    this.data = error
   if (this.callbacks.length > 0) {
                setTimeout(() => {
                    this.callbacks.forEach(callbacksObj => {
                        callbacksObj.onRejected(error)
                    })
                })
            }
}

三、重头戏开始,写原型对象then方法,Promise的结果根据执行的结果返回

Promise.prototype.then = (onResolved,onRejected) => {
   return new Promise((resolve,reject) => {
      function handle(callback){
           /*  
              1、如果抛出异常,return的promise就会失败
                2、如果回调函数返回的不是promise,return就会成功
                 3、如果回调函数返回的是promise,return的promise结果就是这个promise的结果
           */
            try {
               const result = callback(self.data)
               if (result instanceof Promise) {  //instanceof 运算符用来检测 Promise.prototype 是否存在于参数 result 的原型链
                    result.then(
                          value => {resolve(value)},
                           error => {reject(error)},
                     )
                            //result.then(resolve, reject)
                     } else {
                           resolve(result)
                     }
                     } catch (e) {
                        reject(e)
                     }
      }

      if(this.status === PENDING){
                 this.callbacks.push({
                        onResolved: () => {
                            handle(onResolved)
                        },
                        onRejected: () => {
                            handle(onRejected)
                        }
                    })
      } else if(this.status === RESOLVED){
          setTimeout(() => {
              handle(onRejected)
          })
      } else if(this.status === REJECTED){
          setTimeout(() => {
              handle(onResolved)
          })
      }
   })
}

四、写原型对象catch方法

 Promise.prototype.catch = function (onRejected) {
            return this.then(undefined,onRejected)
        }
上一篇 下一篇

猜你喜欢

热点阅读