核心

构造函数核心

  • 维护状态变量,只能由pending变为resolve或者reject
  • 维护一个存储结果的变量
  • 维护一个回调数组,执行到then,如果我们传入的立即执行函数没有立即执行resolve或者reject,所以promise的状态还是pending,这时要把then里面的回调函数保存起来。待到resolve或者reject执行后则执行回调数组里存到方法。若传入的立即执行函数直接执行了resolve或者reject此时就不用把回调保存起来,直接执行onResolved或onRejected方法。注意是异步执行。而且是做为微任务的。(下面我们用setTimeout模拟)

then核心

  • 执行时,若当前状态为pending,则把回调保存到回调数组,若为resolve或者reject则直接异步执行(微任务)。如果回调函数返回的不是promise,return的promise的状态是resolved,value就是返回的值。即resolve(result)或者reject(result)
  • 每次then都返回一个新的Promise。
  • promise会发生值传透,当then中传入的不算函数,则这个then返回的promise的data,将会保存上一个的promise.data。这就是发生值穿透的原因。而且每一个无效的then所返回的promise的状态都为resolved。

核心实现

将Promise向外暴露

  1. (function (window) {
  2. /*
  3. Promise构造函数
  4. executor:执行器函数
  5. */
  6. function Promise(executor) {
  7.  
  8. }
  9.  
  10. // 向外暴露Promise
  11. window.Promise = Promise
  12. })()

构造函数实现

  1. function MyPromise(exector) {
  2. var self = this;
  3. self.status = 'pending'; // 给promise对象指定status属性,初始值为pending
  4. self.data = undefined; // 给promise对象指定一个存储结果的data
  5. self.callbacks = []; // 每个元素的结构 {onResolved(){},onRejected(){}}
  6. function resolve(value) {
  7. if(self.status !== 'pending') {
  8. return
  9. }
  10. self.status = 'resolved'; // 将状态更改为resolved
  11. self.data = value; // 保存value的值
  12. // 异步调用resolve才会在这里执行
  13. // 如果有待执行的callback函数,立即异步执行回调函数onResolved();
  14. if(self.callbacks.length > 0) {
  15. self.callbacks.forEach(callbacksObj => {
  16. callbacksObj.onResolved(value);
  17. });
  18. }
  19. }
  20.  
  21. function reject() {
  22. // 如果当前状态不是pending,则不执行
  23. if(self.status !== 'pending'){
  24. return
  25. }
  26. // 将状态改为rejected
  27. self.status = 'rejected';
  28. // 保存value的值
  29. self.data = value;
  30. // 如果有待执行的callback函数,立即异步执行回调函数onResolved
  31. if (self.callbacks.length>0){
  32. self.callbacks.forEach(callbackObj=>{
  33. callbackObj.onRejected(value)
  34. })
  35. }
  36. }
  37. // 立即同步执行exector
  38. // 注意️:当在执行executor的时候,如果执行异常的话,这个promise的状态会直接执行reject方法
  39. try{
  40. // 立即同步执行executor
  41. executor(resolve,reject);
  42. } catch (e) {
  43. // 如果执行器抛出异常,promise对象变为rejected状态
  44. reject(e);
  45. }
  46. }

then实现

  1. MyPromise.prototype.then = function(onResolved, onReject) {
  2. var self = this;
  3. // 处理值穿透
  4. onResolved = typeof onResolved === 'function'? onResolved: value => value
  5. onRejected = typeof onRejected === 'function'? onRejected: reason => {throw reason}
  6.  
  7. return new myPromise((resolve,reject) => {
  8. if(self.status === 'pending') {
  9. // promise当前状态还是pending状态,将回调函数保存起来
  10. self.callbacks.push({
  11. onResolved() {
  12. handle(onResolved)
  13. },
  14. onRejected() {
  15. handle(onRejected)
  16. }
  17. })
  18. } else if (self.status === 'resolved') {
  19. setTimeout(()=>{
  20. handle(onResolved)
  21. })
  22. } else { // 当status === 'rejected'
  23. setTimeout(()=>{
  24. handle(onRejected)
  25. })
  26. }
  27. // 处理函数
  28. function handle(callback) {
  29. try {
  30. const result = callback(self.data)
  31. if (result instanceof MyPromise){
  32. // 如果回调函数返回的是promise,return的promise的结果就是这个promise的结果
  33. result.then(
  34. value => {resolve(value)},
  35. reason => {reject(reason)}
  36. )
  37. } else {
  38. // 如果回调函数返回的不是promise,return的promise的状态是resolved,value就是返回的值。
  39. resolve(result)
  40. }
  41. } catch (e) {
  42. // 如果执行onResolved的时候抛出错误,则返回的promise的状态为rejected
  43. reject(e)
  44. }
  45. }
  46. });
  47. }

其他方法实现

catch

借用then方法

  1. MyPromise.prototype.catch = function(onRejected){
  2. return this.then(undefined,onRejected)
  3. }

finally

借用then方法

  1. MyPromise.prototype.finally = (onFinally) => {
  2. return this.then((res)=>{
  3. MyPromise.resolve(onFinally()).then(()=> res)
  4. },(reson)=>{
  5. MyPromise.resolve(onFinally()).then(()=> reson)
  6. })
  7. }

resolve

Promise参数可以为如下三种

  • 不是promise
  • 成功状态的promise
  • 失败状态的promise
  1. MyPromise.resolve = function(value){
  2. return new MyPromise((resolve,reject)=>{
  3. if (value instanceof Promise){
  4. // 如果value 是promise
  5. value.then(
  6. value => {resolve(value)},
  7. reason => {reject(reason)}
  8. )
  9. } else{
  10. // 如果value不是promise
  11. resolve(value)
  12. }
  13. }
  14. }

reject

  1. MyPromise.reject = function(reason) {
  2. return new MyPromise((resolve,reject)=>{
  3. reject(reason)
  4. })
  5. }

all

  • 入参一般是个由Promise实例组成的数组,但是也可以不是数组,但必须具有 Iterator 接口,且返回的每个成员都是 Promise 实例。若参数如果不是 Promise 实例,就会先调用Promise.resolve()方法,将参数转为 Promise 实例,再进一步处理。
  • 返回值是个promise,因为可以使用.then
  • 如果全部成功,状态变为resolved, 并且返回值组成一个数组传给回调
  • 但凡有一个失败,状态变为rejected, 并将error返回给回调
  1. MyPromise.all = (promisesArr) => {
  2. // 返回Promise
  3. return new MyPromise((resolve, reject) => {
  4. let dataArr = new Array(promisesArr.length);
  5. let count = 0;
  6. for (let i = 0; i < promisesArr.length; i++) {
  7. // 在 .then 中收集数据,并添加 .catch,在某一个 Promise 遇到错误随时 reject。
  8. // 这样,在最外面调用 Promise.all().catch() 时也可以 catch 错误信息
  9.     // 判断当前这个元素是否为Promise对象,不是则转为Promise对象
  10.    let currentPromise = (promisesArr[i] instanceof Promise) ? promisesArr[i] : Promise.resolve(promisesArr[i]);
  11. currentPromise.then(res => {
  12.     dataArr[index] = data;
  13. count++;
  14. // 如果数据收集完了,就把收集的数据 resolve 出去
  15. if (count === promisesArr.length) resolve(dataArr);
  16.   }).catch(err => {
  17.     //如果某一个失败,promise.all()立即执行reject回调。
  18. //但剩余的promise依旧继续执行,只不过对promise.all的结果不会产生影响了
  19.     reject(err)
  20.   });
  21. }
  22. })

注意️:dataArr添加时用下标而不用数组时为了防止顺序错乱

race

返回一个promise对象,状态由第一个完成的promise决定

  1. MyPromise.race = function(promisesArr){
  2. return new Promise((resolve,reject)=>{
  3. // 遍历promises,获取每个promise的结果
  4. for (let i = 0; i < promisesArr.length; i++) {
  5. // 判断当前这个元素是否为Promise对象,不是则转为Promise对象
  6. let currentPromise = (promisesArr[i] instanceof Promise) ? promisesArr[i] : Promise.resolve(promisesArr[i]);
  7. currentPromise.then(
  8. value => {
  9. // 只要有一个成功,返回的promise的状态九尾resolved
  10. resolve(value)
  11. },
  12. reason => { //只要有一个失败,return的promise状态就为reject
  13. reject(reason)
  14. }
  15. )
  16. })
  17. })
  18. }

allSettled

Promise.allSettled() 方法返回一个在所有给定的 promise 已被决议或被拒绝后决议的 promise,并带有一个对象数组,每个对象表示对应的promise 结果。

  1. Promise.newAllSettled = function (promisesArr) {
  2. return new Promise((resolve, reject) => {
  3. let results = [];
  4. let count = 0;
  5. let promisesArrLength = promisesArr.length;
  6. // 运行所有的 Promise
  7. for (let i = 0; i < promisesArr.length; i++) {
  8. // 判断当前这个元素是否为Promise对象,不是则转为Promise对象
  9. let currentPromise = (promisesArr[i] instanceof Promise) ? promisesArr[i] : Promise.resolve(promisesArr[i]);
  10. currentPromise.then(res => {
  11. // 当有 Promise 被 resolve 之后,记录 resolve 值和状态,已决 Promise 计数加一
  12. results.push({value: res, status: 'fulfilled'});
  13. count++;
  14. // 全部 Promise 已决,resolve
  15. if (count === promisesArrLength) {
  16. resolve(results);
  17. }
  18. }).catch(err => {
  19. // 当有 Promise 被 reject 后,记录 reject 值和状态,并且已决的 Promise 计数加一
  20. results.push({value: err, status: 'rejected'});
  21. count++;
  22. if (count === promisesArrLength) {
  23. resolve(results);
  24. }
  25. });
  26. }
  27. })
  28. };

参考

https://juejin.im/post/6844904088963022856

https://juejin.im/post/6850037281206566919

https://blog.csdn.net/MichelleZhai/article/details/104475521

https://zhuanlan.zhihu.com/p/60287801

https://zhuanlan.zhihu.com/p/41502945

https://zhuanlan.zhihu.com/p/61681036

https://juejin.im/post/6856213486633304078

https://zhuanlan.zhihu.com/p/107712626

Promise核心实现的更多相关文章

  1. Promise核心原理解析

    作者: HerryLo 本文永久有效链接: https://github.com/AttemptWeb...... Promises对象被用于表示一个异步操作的最终完成 (或失败), 及其结果值.主要 ...

  2. promise核心6 自定义promise

    1.定义整体结构(不写实现) 定义一个自己的promise的库 lib(库的简写) 一个js文件.一个js模块(不能用es6  也不能commjs)(用es5模块语法 ) 匿名函数自调用.IIFE ( ...

  3. promise 核心 几个小问题

    1.如何改变pending的壮体 抛出异常.pending变为rejected  // throw new Error('fail')  内部抛出异常也这样 reason为抛出的error resol ...

  4. promise核心 为什么用promise

    为什么要用promise 1.使用纯回调函数 先指定回调函数,再启动异步任务 答 1.指定回调函数的方式更加灵活 可以在执行任务前,中,后 2.支持链式调用,解决回调地狱问题 什么是回调地狱:回调函数 ...

  5. Promise核心基础

    基础 Promise 抽象表达:是js中进行异步编程的新的解决方案 具体解释:1.从语法上来说是一个构造函数 2.从功能上来说promise对象用来封装一个异步操作并可以获取其结果 状态改变:0.ne ...

  6. javascript基础修炼(7)——Promise,异步,可靠性

    开发者的javascript造诣取决于对[动态]和[异步]这两个词的理解水平. 一. 别人是开发者,你也是 Promise技术是[javascript异步编程]这个话题中非常重要的,它一度让我感到熟悉 ...

  7. 分步理解 Promise 的实现

    一个 Promise 的运用: var firstPromise = new Promise(function(resolve,reject){ setTimeout(function(){ var ...

  8. 前端面试?这份手撸Promise请你收下

    前言 现在很多大厂面试前端都会要求能够手动的写出一个Promise,所以这里整理了一份手写的Promise. 绝对详细,功能绝对强大.如果你不了解Promise的基本使用,那么本篇文章可能不太适合你, ...

  9. Generator库co4.6使用及源码分析

    原文链接 http://www.cnblogs.com/ytu2010dt/p/6043947.html co4.x已经抛弃了原来thunk转而结合promise实现. 一:promise proms ...

随机推荐

  1. 2020-06-18:ZK的分布式锁怎么实现?

    福哥答案2020-06-18: Zk分布式锁有两种实现方式一种比较简单,应对并发量不是很大的情况.获得锁:创建一个临时节点,比如/lock,如果成功获得锁,如果失败没获得锁,返回false释放锁:删除 ...

  2. C#LeetCode刷题-字符串

    字符串篇 # 题名 刷题 通过率 难度 3 无重复字符的最长子串   24.6% 中等 5 最长回文子串   22.4% 中等 6 Z字形变换   35.8% 中等 8 字符串转整数 (atoi)   ...

  3. notepad++ 设置支持golang语法高亮

    see https://stackoverflow.com/questions/27747457/golang-plugin-on-notepad

  4. HBase存储及读写原理介绍

    一.HBase介绍及其特点 HBase是一个开源的非关系型分布式数据库,它参考了谷歌的BigTable建模,实现的编程语言为Java.它是Apache软件基金会的Hadoop项目的一部分,运行于HDF ...

  5. 攻防世界-web(进阶)-NaNNaNNaNNaN-Batman

    用winhex打开,发现是一个javascript代码.将文件重命名为html文件,用浏览器打开. 打开是一个输入框,输入任何东西都梅反应,尝试弹框输入也无果,继续查看代码. 查看代码,可以看到最开始 ...

  6. 兼容ie9上传本地资源

    在ie9中上传文件出现问题,大多数的上传文件都采用new Formdata创建添加文件,在IE9中不支持Formdata对象操作,ie10是支持的.所以只能使用表单提交的方式进行操作. <for ...

  7. JavaScript学习系列博客_5_JavaScript中的强制类型转换

    -强制类型转换为String 1.方式1 调用被转换数据的toString()方法 number类型值.布尔类型值.都可以调用toString()方法强制转换.但是null值和undefined值不行 ...

  8. C++ Templates (1.6 但是为什么不...? But, Should't We ...?)

    返回完整目录 目录 1.6 但是为什么不...? But, Should't We ...? 1.6.1 传值还是传引用? Pass by Value or by Reference? 1.6.2 为 ...

  9. springMVC入门(四)------参数绑定与返回值类型

    简介 从之前的介绍,已经可以使用springMVC完成完整的请求.返回数据的功能. 待解决的问题:如何将数据传入springMVC的控制器进行后续的处理,完成在原生servlet/jsp开发中Http ...

  10. Sqlalchemy 事件监听与初始化

    sqlalchemy不仅仅能自动创建数据库,更提供了其他更强大的功能,今天要介绍的就是sqlalchemy中的事件监听,并将其应用到数据库的初始化中. 需求:当插入设置password字段时,自动加密 ...