手写promise,promise.all,promise.race

Posted Mpixer

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了手写promise,promise.all,promise.race相关的知识,希望对你有一定的参考价值。

手写promise

  //手写promise
  const RESOLVE = 'resolved';//成功状态
  const REJECT = 'rejected'//失败状态
  const PENDING = 'Pending'//等待状态

  class MyPromise {
   status = PENDING;
   result = undefined
   reason = undefined
    onResolvedArr = []
    onRejectArr = []
   constructor(excution) {//promise中传入一个函数,我们定义一个excution函数
     //定义resolve与reject两个参数
     const resolve =(result) =>{//resolve中传入参数result
       if(this.status === PENDING){//promise中状态改变不可逆向,只能从PENDING转为RESOLVE或者PENDING转为REJECT
         this.result = result
         this.status = RESOLVE
         this.onResolvedArr.map((fn)=>fn())
       }
     }
     const reject =(reason) =>{//reject传入参数reason
       if (this.status === PENDING){
         this.reason = reason
         this.status = REJECT
         this.onRejectArr.map((fn)=>fn())
       }

     }
     excution(resolve,reject)//传入reslove,reject 2个参数
   }
    //传入promise的then方法
   then(onResoved,onRejected){
     if (this.status === RESOLVE){
       //利用settimeout把任务变为异步任务
       setTimeout(()=>{
         onResoved(this.result)
       },0)

     }
     if (this.status === REJECT){
       setTimeout(()=>{
         onRejected(this.reason)
       },0)
     }
     //加入promise中传入一个异步的函数,通过发布订阅模式来实现监听
     if(this.status === PENDING){
       this.onResolvedArr.push(() => {
         onResoved(this.result)
       })
       this.onRejectArr.push(() => {
         onRejected(this.reason)
       })
     }
   }
  }
//应用
  const  test = new MyPromise((resolve,reject)=>{
    setTimeout(()=>{
      resolve('hello world')
    },1000)
  })
  test.then((res) => {
    console.log(res)
  })
  console.log(1111)

手写promise.all

  Promise.all = function (iterator) {//promise.all传入的为一个迭代器
    const promises = Array.from(iterator)//将迭代器转化成数组
    let count = 0
    let res = []//储存结果
    return new Promise((resolve,reject)=>{//promise.all返回一个promise对象
      for (let i in promises){//对传入的数据进行遍历
        Promise.resolve(promises[i])//将传入的数据每个都转化成promise对象
        .then((data)=>{
          res[i] = data//调用成功存入res中
          if (++count === promises.length){//每次调用成功累加count,当count等于promises的长度,就表明每次都调用成功
            resolve(res)//则返回resolve
          }
        })
        .catch((e)=>{//当count不等于promises的长度则表示其中有失败的调用,则放回reject
          reject(e)
        })
      }
    })
  }

手写promise.race(与all同理,不需要判断)

     Promise.race = function (iterator) {
    const promises = Array.from(iterator)
    return new Promise((resolve,reject) => {
      for (let i in promises) {
        Promise.resolve(promises[i])
            .then((res) => {
              resolve(res)
            })
            .catch(e => {
              reject(e)
            })
      }

    })
  }

以上是关于手写promise,promise.all,promise.race的主要内容,如果未能解决你的问题,请参考以下文章

手写promise.all和race

因为实现不了Promise.all,一场面试凉凉了

Promise的特性

promiseall退出条件

promise.all 解说

如何使用 Promise.all 使用 jest 为多个提取设置测试