0


2022年了--你还不会手写promise? 完成promise的所有实现

手写promise,将静态方法和实例方法都进行了一遍实现。也可以实现链式调用。

  1. /*
  2. * @Author: 毛毛
  3. * @Date: 2022-01-03 10:17:47
  4. * @Last Modified by: 毛毛
  5. * @Last Modified time: 2022-01-03 10:17:47
  6. *//**
  7. *
  8. * 手写promise
  9. * @class MaoPromise
  10. */classMaoPromise{/**
  11. * 正在执行的状态
  12. *
  13. * @static
  14. * @memberof MaoPromise
  15. */static _PROMISE_STATUS_PENDING ="pending";/**
  16. * 成功执行的状态
  17. *
  18. * @static
  19. * @memberof MaoPromise
  20. */static _PROMISE_STATUS_FULFILLED ="fulfilled";/**
  21. * 失败执行的状态
  22. *
  23. * @static
  24. * @memberof MaoPromise
  25. */static _PROMISE_STATUS_REJECTED ="rejected";/**
  26. * 默认的状态 执行中
  27. *
  28. * @memberof MaoPromise
  29. */
  30. _status = MaoPromise._PROMISE_STATUS_PENDING;/**
  31. * 成功执行时 传给 resolve函数的参数
  32. *
  33. * @memberof MaoPromise
  34. */
  35. _value =undefined;/**
  36. * 失败执行时 传给 reject函数的参数
  37. *
  38. * @memberof MaoPromise
  39. */
  40. _reason =undefined;/**
  41. * 成功执行的回调函数
  42. *
  43. * @memberof MaoPromise
  44. */
  45. _onFulfilledCallback =[];/**
  46. * 失败执行的回调函数
  47. *
  48. * @memberof MaoPromise
  49. */
  50. _onRejectedCallback =[];/**
  51. * Creates an instance of MaoPromise.
  52. * @param {Function} executor 执行器
  53. * @memberof MaoPromise
  54. */constructor(executor){try{executor(this.resolve,this.reject);}catch(err){this.reject(err);}}/**
  55. * 静态方法resolve,返回一个执行成功回调的promise对象
  56. *
  57. * @static
  58. * @param {*} value 执行成功回调resolve的参数
  59. * @return {MaoPromise}
  60. * @memberof MaoPromise
  61. */staticresolve(value){returnnewMaoPromise((resolve)=>resolve(value));}/**
  62. * 静态方法reject 返回一个指向reject回调的promise对象
  63. *
  64. * @static
  65. * @param {*} reason 执行reject回调的参数
  66. * @return {MaoPromise}
  67. * @memberof MaoPromise
  68. */staticreject(reason){returnnewMaoPromise((resolve, reject)=>reject(reason));}/**
  69. *
  70. * 静态方法 执行promise数组内的全部的promise,全都执行完 且都是成功执行回调
  71. * 则执行 resolve回调,且参数为全部成功执行promise元素的回调函数resolve的参数的数组。一旦有一个promise元素执行了reject或者抛出异常,则终止执行且立刻执行reject回调
  72. * @static
  73. * @param {Array<MaoPromise>} promises 是一个promise数组
  74. * @return {*}
  75. * @memberof MaoPromise
  76. */staticall(promises){returnnewMaoPromise((resolve, reject)=>{const values =[];
  77. promises.forEach(promise=>{
  78. promise.then(res=>{
  79. values.push(res);if(values.length === promises?.length)resolve(values);},err=>reject(err));});});}/**
  80. *
  81. * allSettled 返回一个在所有给定的promise都已经fulfilled或rejected后的promise,并带有一个对象数组,每个对象表示对应的promise结果。
  82. * @static
  83. * @param {Array<MaoPromise>} promises 是一个promise数组
  84. * @return {*}
  85. * @memberof MaoPromise
  86. */staticallSettled(promises){returnnewMaoPromise((resolve)=>{const result =[];
  87. promises.forEach(promise=>{
  88. promise.then(res=>{
  89. result.push({
  90. status: MaoPromise._PROMISE_STATUS_FULFILLED,
  91. value: res
  92. });if(result.length === promises?.length)resolve(result);},err=>{
  93. result.push({
  94. status: MaoPromise._PROMISE_STATUS_REJECTED,
  95. reason: err
  96. });if(result.length === promises?.length)resolve(result);});})});}/**
  97. *
  98. * promise数组的元素 有一个promise拿到了结果,无论是成功还是失败,都直接结束
  99. * @static
  100. * @param {Array<MaoPromise>} promises 是一个promise数组
  101. * @memberof MaoPromise
  102. */staticrace(promises){returnnewMaoPromise((resolve, reject)=>{
  103. promises.forEach(promise=>{
  104. promise.then(resolve, reject);});});}/**
  105. *
  106. * 只要其中的一个 promise 成功,就返回那个已经成功的 promise 。
  107. * 如果可迭代对象中没有一个 promise 成功(即所有的 promises 都失败/拒绝),
  108. * 就返回一个失败的 promise 和AggregateError类型的实例,
  109. * 它是 Error 的一个子类,用于把单一的错误集合在一起。
  110. * 本质上,这个方法和Promise.all()是相反的。
  111. * @static
  112. * @param {Array<MaoPromise>} promises 是一个promise数组
  113. * @return {*}
  114. * @memberof MaoPromise
  115. */staticany(promises){returnnewMaoPromise((resolve, reject)=>{const reasons =[];
  116. promises.forEach(promise=>{
  117. promise.then(resolve,err=>{
  118. reasons.push(err);if(reasons.length === promises?.length)reject(newAggregateError(reasons));});});});}/**
  119. * 成功时执行
  120. *
  121. * @param {*} value
  122. * @memberof MaoPromise
  123. */resolve=(value)=>{if(this._status === MaoPromise._PROMISE_STATUS_PENDING){// 延迟执行 queueMicrotask函数 将回调函数的内容加入到微任务中执行queueMicrotask(()=>{if(this._status !== MaoPromise._PROMISE_STATUS_PENDING)return;this._value = value;this._status = MaoPromise._PROMISE_STATUS_FULFILLED;// 执行成功回调this._onFulfilledCallback.forEach(callback=>{callback(this._value);});});}}/**
  124. * 失败时执行
  125. *
  126. * @param {*} reason
  127. * @memberof MaoPromise
  128. */reject=(reason)=>{if(this._status === MaoPromise._PROMISE_STATUS_PENDING){queueMicrotask(()=>{if(this._status !== MaoPromise._PROMISE_STATUS_PENDING)return;this._reason = reason;this._status = MaoPromise._PROMISE_STATUS_REJECTED;// 执行失败回调this._onRejectedCallback.forEach(callback=>{callback(this._reason);});});}}/**
  129. * then方法
  130. *
  131. * @param {*} onFulfilled 成功回调
  132. * @param {*} onRejected 失败回调
  133. * @memberof MaoPromise
  134. */then(onFulfilled, onRejected){// 如果 onRejected函数没有传 想要在catch方法中传回调// TODO 那么如果传入了onRejected回调,又使用catch进行捕获会如何?
  135. onRejected = onRejected ??(err=>{throw err });// 如果第一个then的resolve函数有返回值,且链式调用过程后面出现的是catch// 则成功回调函数是 undefined,也就是返回值不会被处理// 所以我们需要在调用catch的时候,将上一个resolve的结果返回出去// 如果不给 onFulfilled赋值,则catch后面链式调用里面的回调函数都不会执行
  136. onFulfilled = onFulfilled ??(value=> value);returnnewMaoPromise((resolve, reject)=>{// TODO 执行then函数的时候,状态已经确定了,则直接执行成功回调函数if(this._status === MaoPromise._PROMISE_STATUS_FULFILLED){if(typeof onFulfilled ==="function"){this._executorFunctionWithCatchError(onFulfilled,this._value, resolve, reject);}}elseif(this._status === MaoPromise._PROMISE_STATUS_REJECTED){if(typeof onRejected ==="function"){this._executorFunctionWithCatchError(onRejected,this._reason, resolve, reject);}}else{// pending 状态// TODO 副作用函数的返回值 作为then函数返回值promise的(resolve,reject)的参数// 状态还没确定之前 搜集副作用 在状态改变之后 一起执行if(typeof onFulfilled ==="function")// 为了收集到副作用执行后的返回值 我们将副作用函数放到新的函数中 然后加入到副作用数组中this._onFulfilledCallback.push(()=>{this._executorFunctionWithCatchError(onFulfilled,this._value, resolve, reject);});if(typeof onRejected ==="function")this._onRejectedCallback.push(()=>{this._executorFunctionWithCatchError(onRejected,this._reason, resolve, reject);});}});}/**
  137. * catch方法的设计 巧妙的用了then方法,
  138. * 但是考虑到我们可能会在catch方法后面,
  139. * 链式的调用finally方法,所以需要将调用的then方法的返回值 继续返回
  140. *
  141. * @param {*} onRejected 失败/异常处理回调
  142. * @memberof MaoPromise
  143. */catch(onRejected){returnthis.then(undefined, onRejected);}/**
  144. * 最终执行promise的善后工作的代码
  145. *
  146. * @param {*} onFinally 最终回调
  147. * @memberof MaoPromise
  148. */finally(onFinally){// 还是借用then方法,不管成功还是失败/异常 都会执行最终回调if(typeof onFinally ==="function")this.then(()=>{onFinally();},()=>{onFinally();});}/**
  149. * 执行副作用函数 进行异常的捕获处理
  150. *
  151. * @param {*} execFn 副作用函数
  152. * @param {*} value 上一个回调函数(resolve,reject)执行时传入的参数
  153. * @param {*} resolve 成功回调
  154. * @param {*} reject 失败回调
  155. * @memberof MaoPromise
  156. */_executorFunctionWithCatchError(execFn, value, resolve, reject){try{const res =execFn(value);resolve(res);}catch(err){reject(err);}}}// MaoPromise.reject("err or").catch((err) => {// console.log(err);// }).finally(() => {// console.log("finally");// });const p1 =newMaoPromise((resolve, reject)=>{setTimeout(()=>{resolve(111);},1000);});const p2 =newMaoPromise((resolve, reject)=>{setTimeout(()=>{reject(222);},2000);});const p3 =newMaoPromise((resolve, reject)=>{setTimeout(()=>{resolve(333);},3000);});
  157. MaoPromise.any([p2]).then((res)=>{
  158. console.log(res);}).catch(err=>{
  159. console.log(err.errors);})

本文转载自: https://blog.csdn.net/weixin_45747310/article/details/122284490
版权归原作者 听雨少年毛毛 所有, 如有侵权,请联系我们删除。

“2022年了--你还不会手写promise? 完成promise的所有实现”的评论:

还没有评论