【手写 Promise 源码】第十二篇 - Promise.race 的实现

news2025/1/9 15:29:40

一,前言

上一篇,主要实现 Promise 的核心静态 API(类方法):Promise.all,主要涉及以下几个点:

  • 测试原生 Promise.all 的使用;
  • Promise.all 的功能与特性分析;
  • Promise.all 的源码实现、执行分析、功能测试;

本篇,继续实现 Promise 静态 API(类方法):Promise.race;

备注:

  • Promise.race 与 Promise.all 实现原理相似;有了 all 的基础 race 就非常简单了;
  • Promise.race 面试高频考察点,主要用于解决超时中断问题;

二,Promise.race 简介

1,API 介绍

race:译为“比赛”、“赛跑”;MDN 参考资料

image.png

Promise.race功能(与 Promise.all 对比):

Promise.all

  • 批处理 promise,返回 promise;
  • 全部成功才成功,返回全部结果;
  • 一个出错就失败,返回第一个失败结果;

Promise.race

  • 批处理 promise,返回 promise;
  • 无论成功还是失败,谁快就返回谁;

2,原生 Promise.race 功能测试

let p1 = new Promise((resolve, reject) => {
  setTimeout(() => {
    console.log("Promise 执行完成:成功")
    resolve('success')
  }, 1000);
})

let p2 = new Promise((resolve, reject) => {
  setTimeout(() => {
    console.log("Promise 执行完成:失败")
    reject('fail')
  }, 3000);
})

Promise.race([p1, p2]).then((data) => {
  console.log('then', data);
}).catch((err) => {
  console.log('catch', err);
})

// 执行结果:
// 1 秒后打印“Promise 执行完成:成功” + “then success”;
// 3 秒后打印“Promise 执行完成:失败”;

备注:

  • 虽然Promise.race在 1 秒后就返回了结果,但实际一共执行了 3 秒才完成;
  • 如果数组传入数字,会直接返回成功;

三,Promise.race 实现

1,原理分析

  • Promise.race:入参是一个 promise 集合;返回一个 Promise 实例;
    Promise.race 内部 return new Promise(...)
  • 执行所有 promise,使用最先返回的结果(不管成功还是失败);
    p.then(resolve, reject) 就看谁先进入 then;

2,代码实现

  static race (promises) {
    return new Promise((resolve, reject)=>{
      for(let i = 0; i < promises.length; i++){
        let p = promises[i];
        if(p && typeof p.then === 'function'){
          p.then(resolve, reject) // 使用第一个执行成功的结果
        }else{
          resolve(p)
        }
      }
    })
  }

3,功能测试

传入数字、成功/失败 promise,表现与原生 promise 一致


四,Promise.race 应用:解决超时问题

借助 Promise.race 特性,为 promise 绑个炸弹:

// 包装 promise,使之具备新功能:控制 promise 状态
function wrap(p1){
  let abort;
  let p = new Promise((resolve, reject)=>{
    // 拿到 p 的 reject,以便于随时更新 p 的状态
    abort = reject; 
  })
  
  // Promise.race:相当于给 p1 绑了一个定时炸弹 p
  let p2 = Promise.race([p, p1]);
  p2.abort = abort;// p2 就获得炸弹的遥控器 
                    
  return p2;
}

超时了,就直接引爆:

let p1 = new Promise((resolve, reject)=>{
  abort = reject;
  setTimeout(() => {
    resolve('成功')
  }, 3000);
})

let p2 = wrap(p1);

p2.then((data)=>{
  console.log('then', data)
},(err)=>{
  console.log('err', err)
})
setTimeout(() => {
   p2.abort('已超时')
}, 1000);

// 执行结果:1 秒后输出"err 已超时",3 秒全部执行完成

五,结尾

本篇,主要实现了 Promise 静态 API(类方法):Promise.race,主要涉及以下几个点:

  • 测试原生 Promise.race 的使用;
  • Promise.race 的功能与特性分析;
  • Promise.race 的源码实现、执行分析、功能测试;

下一篇,继续 Promise 静态 API:Promise.allSettledPromise.any


维护记录

  • 20211106
    • 添加 Promise.race 应用;(仍需完善)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/189686.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

什么是3D点云数据?该如何标注它?| 数据标注

自动驾驶汽车严重依赖输入数据来做出驾驶决策。一般来讲&#xff0c;数据越详细&#xff0c;车辆做出的决策就越好车辆行驶的也就更安全。虽然现代相机可以捕获非常详细的数据表示&#xff0c;但输出仍然是2D的&#xff0c;它限制了我们可以提供给操作车辆的神经网络的信息。相…

Java 技术知识点汇总

背景 面向百度编程&#xff0c;基本能解决大部分的问题。然而&#xff0c;技术用的熟练&#xff0c;时间久了却容易遗忘背后的技术思想&#xff0c;倒成了妥妥的搬运工了。 所以&#xff0c;偶尔针对用的技术&#xff0c;发几个灵魂拷问&#xff0c;也是很有必要的&#xff0…

Linux 软件管理 YUM管理工具 配置本地YUM源

概念引入 &#xff1a; # 首先提出一个问题&#xff0c;我们在 Linux 操作系统中是如何 安装软件的 &#xff1f;&#xff1f; >>> 在 Linux 系统中&#xff0c;安装软件是有三种方式 >>> 第一种 &#xff1a; RPM 管理工具 第二种 &#xff1a; YUM 管理工…

11、矩阵的分解

目录 一、对称正定矩阵的Cholesky分解 二、一般方阵的高斯消去法分解 三、矩形矩阵的正交分解 四、舒尔分解 矩阵分解是把一个矩阵分解成几个“较简单”的矩阵连乘的形式。在MATLAB中矩阵分解的相关函数有&#xff1a; 在MATLAB中&#xff0c;线性方程组的求解主要基于4种基…

剑指 Offer 第15天 搜索与回溯算法(中等)

目录 剑指 Offer 34. 二叉树中和为某一值的路径 剑指 Offer 36. 二叉搜索树与双向链表 剑指 Offer 54. 二叉搜索树的第k大节点 剑指 Offer 34. 二叉树中和为某一值的路径 给你二叉树的根节点 root 和一个整数目标和 targetSum &#xff0c;找出所有 从根节点到叶子节点 路径…

【懒狗福音】设置Bios电源自启实现远程办公

目录前言正文需求分析开始实现需求1需求2总结前言 今年寒假升级了下家里台式机的配置&#xff0c;已经很能满足我的生产力需求。 尽管我是按着MATX主机的方向配的&#xff0c;选择的机箱也是净重较轻的铝合金材质&#xff0c;机身自带提手也还算便捷。 但是装配完毕后它的整…

前端开发:JS中数组常用方法汇总

前言 在前端开发中关于数组的使用想必前端开发者并不陌生&#xff0c;尤其是在处理业务逻辑的时候&#xff0c;从后端获取的数据类型中数组类型基本占到70%的比例&#xff0c;所以与其说是处理数据&#xff0c;不如说是处理数组&#xff0c;虽然说的有点夸张&#xff0c;前端实…

vue文件导出/下载

const blob new Blob([res.data]);const elink document.createElement(a);elink.download 导出数据.xlsx;elink.style.display none;elink.href URL.createObjectURL(blob);document.body.appendChild(elink);elink.click();URL.revokeObjectURL(elink.href);document.bo…

使用Coding管理项目代码记录

直接开门见山 一、创建项目 进入coding登录后&#xff0c;找到项目菜单&#xff0c;然后点击创建项目 二、创建代码仓库 进入项目中&#xff0c;针对不同需要创建代码仓库。大项目或者前后端分离分开开发的可以创建不同的代码仓库。 三、创建代码仓库与本地关联 项目管…

Java岗面试题--Java基础(日积月累,每日三题)

目录面试题一&#xff1a;JDK、JRE、JVM之间的区别面试题二&#xff1a;hashCode()与equals()之间的关系追问&#xff1a;为什么重写 equals() 就一定要重写 hashCode() 方法&#xff1f;面试题三&#xff1a;String、StringBuffer、StringBuilder的区别追问一&#xff1a;Stri…

vulfocus靶场通关(目录遍历)

uWSGI 目录穿越&#xff08;CVE-2018-7490&#xff09; uWSGI是一款Web应用程序服务器&#xff0c;它实现了WSGI、uwsgi和http等协议&#xff0c;并支持通过插件来运行各种语言,uWSGI 2.0.17之前的PHP插件&#xff0c;没有正确的处理DOCUMENT_ROOT检测&#xff0c;导致用户可以…

【运维】通过gotty实现网页代理访问服务器及K8S容器操作实践

Gotty 是Golang编写的可以方便的共享系统终端为web应用&#xff0c;是一个灵活强大的通过web访问终端的工具。本文将主要通过搭建Gotty实现对K8S容器的访问操作&#xff0c;开发如果想要正常的进行容器访问以及测试环境代码调试&#xff0c;最好是搭建一套与运维环境隔离的应用…

Gadget驱动程序框架

Gadget驱动程序框架 文章目录Gadget驱动程序框架参考资料&#xff1a;一、 怎样理解Gadget框架二、从硬件软件角度理解Gadget框架2.1 底层硬件操作_UDC驱动2.2 上层软件操作三、 从构造描述符的角度理解Gadget框架致谢参考资料&#xff1a; Linux下USB gadget设备详解Linux us…

Linux常用命令——quotaon命令

在线Linux命令查询工具(http://www.lzltool.com/LinuxCommand) quotaon 激活Linux内核中指定文件系统的磁盘配额功能 补充说明 quotaon命令用于激活Linux内核中指定文件系统的磁盘配额功能。执行quotaon指令可开启用户和群组的才磅秒年空间限制&#xff0c;各分区的文件系统…

STM32——I2C外设总线

文章目录一、I2C外设简介二、I2C框图三、I2C基本结构四、主机发送五、主机接收六、I2C的中断请求七、软件/硬件波形对比八、硬件I2C读写MPU6050电路设计关键代码状态监控函数一、I2C外设简介 STM32内部集成了硬件I2C收发电路&#xff0c;可以由硬件自动执行时钟生成、起始终止条…

Go语言jwt无感刷新以及解决SSO单点登录限制

前言 为什么使用JWT&#xff1f; Jwt提供了生成token以及token验证的方法&#xff0c;而token是一种不用存储在服务端&#xff0c;只需要由用户携带即可实现认证的一种方式。在介绍JWT之前&#xff0c;我们也应该先了解cookie和session。 Cookie和Session 为每一位用户设定…

PLSQL Developer 代码助手卡顿优化

支持付费优化 原因分析 代码助手卡顿来源于&#xff08;不考虑网络和软件版本等影响&#xff09; A.从已连接的数据库的数据字典中&#xff0c;读取该表的列信息B.将读取到的列信息返回到plsql编辑器&#xff0c;并进行字符处理&#xff0c;然后显示如何优化B 这个勾勾是维…

【GPLT 二阶题目集】L2-034 口罩发放

为了抗击来势汹汹的 COVID19 新型冠状病毒&#xff0c;全国各地均启动了各项措施控制疫情发展&#xff0c;其中一个重要的环节是口罩的发放。 某市出于给市民发放口罩的需要&#xff0c;推出了一款小程序让市民填写信息&#xff0c;方便工作的开展。小程序收集了各种信息&#…

redis的配置文件

目录 介绍 1. 单位的设置方式 2. 可以包含其他文件内容 3. 网络配置相关 4. 通用 5. 安全 6. 限制 redis的配置文件名为redis.conf&#xff0c;一般会安装在/etc目录下。 如果找不到该文件可以使用find命令查找。 find . -name redis.conf 介绍 1. 单位的设置方式 只支持字…

client-go监听apiserver,监听http2逻辑分析

前言 最近做项目&#xff0c;需要写一个controller&#xff08;k8s的插件&#xff09;&#xff0c;需要从k8s的apiserver取数据&#xff0c;就用了自带的client-go&#xff0c;但是client-go是怎么从apiserver获取数据的一直没有研究过&#xff0c;只是看网上&#xff0c;看官…