面试手写实现Promise.all

news2024/11/14 15:06:19
目录
  • 前言
  • 常见面试手写系列
  • Promise.resolve
    • 简要回顾
    • 源码实现
  • Promise.reject
    • 简要回顾
    • 源码实现
  • Promise.all
    • 简要回顾
    • 源码实现
  • Promise.allSettled
    • 简要回顾
    • 源码实现
  • Promise.race
    • 简单回顾
    • 源码实现
  • 结尾

前言

(?﹏?)曾经真实发生在一个朋友身上的真实事件,面试官让他手写一个Promise.all,朋友现场发挥不太好,没有写出来,事后他追问面试官给的模糊评价是基础不够扎实,原理性知识掌握较少... 当然整场面试失利,并不仅仅是这一个题目,肯定还有其他方面的原因。

但是却给我们敲响一个警钟:Promise手写实现、Promise静态方法实现早已经是面试中的高频考题,如果你对其还不甚了解,耽误你10分钟,我们一起干到他懂O(∩_∩)O

常见面试手写系列

最近很想做一件事情,希望可以将前端面试中常见的手写题写成一个系列,尝试将其中涉及到的知识和原理都讲清楚,如果你对这个系列也感兴趣,欢迎一起来学习噢,目前已有66+手写题实现啦!

1. 点击查看日拱一题源码地址(目前已有66+个手写题实现)

2.脚本之家专栏

Promise.resolve

简要回顾

  • Promise.resolve(value) 方法返回一个以给定值解析后的Promise 对象。
  • 如果这个值是一个 promise ,那么将返回这个 promise ;
  • 如果这个值是thenable(即带有"then" 方法),返回的promise会“跟随”这个thenable的对象,采用它的最终状态;否则返回的promise将以此值完成。

这是MDN上的解释,我们挨个看一下

  • Promise.resolve最终结果还是一个Promise,并且与Promise.resolve(该值)传入的值息息相关
  • 传入的参数可以是一个Promise实例,那么该函数执行的结果是直接将实例返回
  • 这里最主要需要理解跟随,可以理解成Promise最终状态就是这个thenable对象输出的值

小例子

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

// 1. 非Promise对象,非thenable对象

Promise.resolve(1).then(console.log) // 1

// 2. Promise对象成功状态

const p2 = new Promise((resolve) => resolve(2))

Promise.resolve(p2).then(console.log) // 2

// 3. Promise对象失败状态

const p3 = new Promise((_, reject) => reject('err3'))

Promise.resolve(p3).catch(console.error) // err3

// 4. thenable对象

const p4 = {

  then (resolve) {

    setTimeout(() => resolve(4), 1000)

  }

}

Promise.resolve(p4).then(console.log) // 4

// 5. 啥都没传

Promise.resolve().then(console.log) // undefined

源码实现

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

Promise.myResolve = function (value) {

  // 是Promise实例,直接返回即可

  if (value && typeof value === 'object' && (value instanceof Promise)) {

    return value

  }

  // 否则其他情况一律再通过Promise包装一下

  return new Promise((resolve) => {

    resolve(value)

  })

}

// 测试一下,还是用刚才的例子

// 1. 非Promise对象,非thenable对象

Promise.myResolve(1).then(console.log) // 1

// 2. Promise对象成功状态

const p2 = new Promise((resolve) => resolve(2))

Promise.myResolve(p2).then(console.log) // 2

// 3. Promise对象失败状态

const p3 = new Promise((_, reject) => reject('err3'))

Promise.myResolve(p3).catch(console.error) // err3

// 4. thenable对象

const p4 = {

  then (resolve) {

    setTimeout(() => resolve(4), 1000)

  }

}

Promise.myResolve(p4).then(console.log) // 4

// 5. 啥都没传

Promise.myResolve().then(console.log) // undefined

疑问

从源码实现中,并没有看到对于thenable对象的特殊处理呀!其实确实也不需要在Promise.resolve中处理,真实处理的地方应该是在Promise构造函数中,如果你对这块感兴趣,马上就会写Promise的实现篇,期待你的阅读噢。

Promise.reject

简要回顾

Promise.reject() 方法返回一个带有拒绝原因的Promise对象。

?

1

2

3

4

5

6

Promise.reject(new Error('fail'))

  .then(() => console.log('Resolved'),

        (err) => console.log('Rejected', err))

// 输出以下内容       

// Rejected Error: fail

//    at <anonymous>:2:16       

源码实现

reject实现相对简单,只要返回一个新的Promise,并且将结果状态设置为拒绝就可以

?

1

2

3

4

5

6

7

8

9

10

11

Promise.myReject = function (value) {

  return new Promise((_, reject) => {

    reject(value)

  })

}

// 测试一下

Promise.myReject(new Error('fail'))

  .then(() => console.log('Resolved'),

        (err) => console.log('Rejected', err))

// Rejected Error: fail

//    at <anonymous>:9:18

Promise.all

简要回顾

Promise.all()方法用于将多个 Promise 实例,包装成一个新的 Promise 实例。这个静态方法应该是面试中最常见的啦

?

1

const p = Promise.all([p1, p2, p3])

最终p的状态由p1、p2、p3决定,分成两种情况。

(1)只有p1、p2、p3的状态都变成fulfilled,p的状态才会变成fulfilled,此时p1、p2、p3的返回值组成一个数组,传递给p的回调函数。

(2)只要p1、p2、p3之中有一个被rejected,p的状态就变成rejected,此时第一个被reject的实例的返回值,会传递给p的回调函数。

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

const p1 = Promise.resolve(1)

const p2 = new Promise((resolve) => {

  setTimeout(() => resolve(2), 1000)

})

const p3 = new Promise((resolve) => {

  setTimeout(() => resolve(3), 3000)

})

const p4 = Promise.reject('err4')

const p5 = Promise.reject('err5')

// 1. 所有的Promise都成功了

const p11 = Promise.all([ p1, p2, p3 ])

    .then(console.log) // [ 1, 2, 3 ]

      .catch(console.log)

// 2. 有一个Promise失败了

const p12 = Promise.all([ p1, p2, p4 ])

    .then(console.log)

      .catch(console.log) // err4

// 3. 有两个Promise失败了,可以看到最终输出的是err4,第一个失败的返回值

const p13 = Promise.all([ p1, p4, p5 ])

    .then(console.log)

      .catch(console.log) // err4

源码实现

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

Promise.myAll = (promises) => {

  return new Promise((rs, rj) => {

    // 计数器

    let count = 0

    // 存放结果

    let result = []

    const len = promises.length

    if (len === 0) {

      return rs([])

    }

    promises.forEach((p, i) => {

      // 注意有的数组项有可能不是Promise,需要手动转化一下

      Promise.resolve(p).then((res) => {

        count += 1

        // 收集每个Promise的返回值

        result[ i ] = res

        // 当所有的Promise都成功了,那么将返回的Promise结果设置为result

        if (count === len) {

          rs(result)

        }

        // 监听数组项中的Promise catch只要有一个失败,那么我们自己返回的Promise也会失败

      }).catch(rj)

    })

  })

}

// 测试一下

const p1 = Promise.resolve(1)

const p2 = new Promise((resolve) => {

  setTimeout(() => resolve(2), 1000)

})

const p3 = new Promise((resolve) => {

  setTimeout(() => resolve(3), 3000)

})

const p4 = Promise.reject('err4')

const p5 = Promise.reject('err5')

// 1. 所有的Promise都成功了

const p11 = Promise.myAll([ p1, p2, p3 ])

    .then(console.log) // [ 1, 2, 3 ]

      .catch(console.log)

// 2. 有一个Promise失败了

const p12 = Promise.myAll([ p1, p2, p4 ])

    .then(console.log)

      .catch(console.log) // err4

// 3. 有两个Promise失败了,可以看到最终输出的是err4,第一个失败的返回值

const p13 = Promise.myAll([ p1, p4, p5 ])

    .then(console.log)

      .catch(console.log) // err4

// 与原生的Promise.all返回是一致的   

Promise.allSettled

简要回顾

有时候,我们希望等到一组异步操作都结束了,不管每一个操作是成功还是失败,再进行下一步操作。显然Promise.all(其只要是一个失败了,结果即进入失败状态)不太适合,所以有了Promise.allSettled

Promise.allSettled()方法接受一个数组作为参数,数组的每个成员都是一个 Promise 对象,并返回一个新的 Promise 对象。只有等到参数数组的所有 Promise 对象都发生状态变更(不管是fulfilled还是rejected),返回的 Promise 对象才会发生状态变更,一旦发生状态变更,状态总是fulfilled,不会变成rejected

还是以上面的例子为例, 我们看看与Promise.all有什么不同

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

const p1 = Promise.resolve(1)

const p2 = new Promise((resolve) => {

  setTimeout(() => resolve(2), 1000)

})

const p3 = new Promise((resolve) => {

  setTimeout(() => resolve(3), 3000)

})

const p4 = Promise.reject('err4')

const p5 = Promise.reject('err5')

// 1. 所有的Promise都成功了

const p11 = Promise.allSettled([ p1, p2, p3 ])

    .then((res) => console.log(JSON.stringify(res, null,  2)))

// 输出

/*

[

  {

    "status": "fulfilled",

    "value": 1

  },

  {

    "status": "fulfilled",

    "value": 2

  },

  {

    "status": "fulfilled",

    "value": 3

  }

]

*/

// 2. 有一个Promise失败了

const p12 = Promise.allSettled([ p1, p2, p4 ])

    .then((res) => console.log(JSON.stringify(res, null,  2)))

// 输出

/*

[

  {

    "status": "fulfilled",

    "value": 1

  },

  {

    "status": "fulfilled",

    "value": 2

  },

  {

    "status": "rejected",

    "reason": "err4"

  }

]

*/

// 3. 有两个Promise失败了

const p13 = Promise.allSettled([ p1, p4, p5 ])

    .then((res) => console.log(JSON.stringify(res, null,  2)))

// 输出

/*

[

  {

    "status": "fulfilled",

    "value": 1

  },

  {

    "status": "rejected",

    "reason": "err4"

  },

  {

    "status": "rejected",

    "reason": "err5"

  }

]

*/

可以看到:

  • 不管是全部成功还是有部分失败,最终都会进入Promise.allSettled的.then回调中
  • 最后的返回值中,成功和失败的项都有status属性,成功时值是fulfilled,失败时是rejected
  • 最后的返回值中,成功含有value属性,而失败则是reason属性

源码实现

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

75

76

77

78

79

80

81

82

83

84

85

86

87

88

89

90

91

92

93

94

95

96

97

98

99

100

101

102

103

104

Promise.myAllSettled = (promises) => {

  return new Promise((rs, rj) => {

    let count = 0

    let result = []

    const len = promises.length

    // 数组是空的话,直接返回空数据

    if (len === 0) {

      return rs([])

    }

    promises.forEach((p, i) => {

      Promise.resolve(p).then((res) => {

        count += 1

        // 成功属性设置

        result[ i ] = {

          status: 'fulfilled',

          value: res

        }

        if (count === len) {

          rs(result)

        }

      }).catch((err) => {

        count += 1

        // 失败属性设置

        result[i] = {

          status: 'rejected',

          reason: err

        }

        if (count === len) {

          rs(result)

        }

      })

    })

  })

}

// 测试一下

const p1 = Promise.resolve(1)

const p2 = new Promise((resolve) => {

  setTimeout(() => resolve(2), 1000)

})

const p3 = new Promise((resolve) => {

  setTimeout(() => resolve(3), 3000)

})

const p4 = Promise.reject('err4')

const p5 = Promise.reject('err5')

// 1. 所有的Promise都成功了

const p11 = Promise.myAllSettled([ p1, p2, p3 ])

    .then((res) => console.log(JSON.stringify(res, null,  2)))

// 输出

/*

[

  {

    "status": "fulfilled",

    "value": 1

  },

  {

    "status": "fulfilled",

    "value": 2

  },

  {

    "status": "fulfilled",

    "value": 3

  }

]

*/

// 2. 有一个Promise失败了

const p12 = Promise.myAllSettled([ p1, p2, p4 ])

    .then((res) => console.log(JSON.stringify(res, null,  2)))

// 输出

/*

[

  {

    "status": "fulfilled",

    "value": 1

  },

  {

    "status": "fulfilled",

    "value": 2

  },

  {

    "status": "rejected",

    "reason": "err4"

  }

]

*/

// 3. 有两个Promise失败了

const p13 = Promise.myAllSettled([ p1, p4, p5 ])

    .then((res) => console.log(JSON.stringify(res, null,  2)))

// 输出

/*

[

  {

    "status": "fulfilled",

    "value": 1

  },

  {

    "status": "rejected",

    "reason": "err4"

  },

  {

    "status": "rejected",

    "reason": "err5"

  }

]

*/

Promise.race

简单回顾

Promise.race()方法同样是将多个 Promise 实例,包装成一个新的 Promise 实例。

?

1

const p = Promise.race([p1, p2, p3])

只要p1、p2、p3之中有一个实例率先改变状态,p的状态就跟着改变。那个率先改变的 Promise 实例的返回值,就传递给p的回调函数。

?

1

2

3

4

5

6

7

8

9

10

11

12

const p1 = new Promise((resolve, reject) => {

  setTimeout(resolve, 500, 1)

})

const p2 = new Promise((resolve, reject) => {

  setTimeout(resolve, 100, 2)

})

Promise.race([p1, p2]).then((value) => {

  console.log(value) // 2

})

Promise.race([p1, p2, 3]).then((value) => {

  console.log(value) // 3

})

源码实现

聪明的你一定马上知道该怎么实现了,只要了解哪个实例先改变了,那么Promise.race就跟随这个结果,那么就可以写出以下代码

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

Promise.myRace = (promises) => {

  return new Promise((rs, rj) => {

    promises.forEach((p) => {

      // 对p进行一次包装,防止非Promise对象

      // 并且对齐进行监听,将我们自己返回的Promise的resolve,reject传递给p,哪个先改变状态,我们返回的Promise也将会是什么状态

      Promise.resolve(p).then(rs).catch(rj)

    })

  })

}

// 测试一下

const p1 = new Promise((resolve, reject) => {

  setTimeout(resolve, 500, 1)

})

const p2 = new Promise((resolve, reject) => {

  setTimeout(resolve, 100, 2)

})

Promise.myRace([p1, p2]).then((value) => {

  console.log(value) // 2

})

Promise.myRace([p1, p2, 3]).then((value) => {

  console.log(value) // 3

})

结尾

也许你我素未谋面,但很可能相见恨晚。希望这里能成为你的栖息之地,我愿和你一起收获喜悦,奔赴成长。

 

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

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

相关文章

网络应用技术师技能考试试题

♥️作者&#xff1a;小刘在C站 ♥️个人主页&#xff1a; 小刘主页 ♥️努力不一定有回报&#xff0c;但一定会有收获加油&#xff01;一起努力&#xff0c;共赴美好人生&#xff01; ♥️学习两年总结出的运维经验&#xff0c;以及思科模拟器全套网络实验教程。专栏&#xf…

2023秋招面试题持续更新中。。。

目录 1.八股文渐进式MVVM三次握手&#xff0c;四次挥手viteajax组件化和模块化虚拟dom原理流程浏览器内核浏览器渲染过程回流和重绘nextTick 2.项目相关1.声明式导航和编程式导航重写push和replace方法&#xff1a;性能优化图片懒加载路由懒加载 http请求方式 1.八股文 渐进式…

怎样快速实现一个发送邮件的接口?flask_mail库30行代码,快速实现

需求 实现了一个根据部门批量获取自动化行覆盖率&#xff0c;并与指标做diff的脚本&#xff0c;怎么才能第一时间通知给对应的人呢&#xff1f; 最先想到的一定是邮件发送(后边我在自己项目中接入的是公司内部的一个类似钉钉的服务号) 因为原本项目是用的flask&#xff0c;所…

DRUPAL 8.x远程代码执行漏洞(CVE-2018-7600)

事件背景 框架漏洞收集 CVE-2018-7600有两个POC分别是7和8的&#xff0c;本文仅研究8版本的POC&#xff0c;与其它的文章不同的事&#xff0c;本文我将数据流向调试并记录下来了 漏洞说明 1. 漏洞原理&#xff1a;Drupal对表单请求内容未做严格过滤&#xff0c;因此&#x…

900就能上8000MHz的DDR5内存,光威神策做到了

这是一款定位于高端国产DDR5电竞游戏内存条的产品&#xff0c;而且是来自国产品牌&#xff0c;频率可以达到8000MHz&#xff0c;设计也是相当惊艳&#xff0c;重点在于它的价格是真的香&#xff0c;有需要的朋友可以冲了。 光威近期发布了一款神策系列DDR5内存条&#xff0c;相…

【腾讯云 Cloud Studio 实战训练营】基于Cloud Studio完成简易通讯录

目录 &#x1f506;Cloud Studio 简介 操作步骤 1.登录 2.创建工作空间 3.初始界面 4.开发空间 5.保存自定义模板 &#x1f506;简易通讯录 1.实验要求 2.操作环境 3.源代码介绍 3.1 定义通讯录类 3.2 定义通讯录列表 3.3 添加联系人功能 3.4 修改联系人 3.5 …

Echarts 柱状图显示百分比

以下是生成的 option option {yAxis: { name: 金额&#xff08;元&#xff09;, type: value },xAxis: { type: category },legend: {},series: [{stack: x,name: 早餐,label: {normal: {show: true,position: insideRight,// 格式化显示formatter: function (params) {let …

清楚知道谁在划水?伙伴云一招搞定任务交办

伙伴云任务交办让每个职场人都拥有专属的事务汇集地&#xff0c; 让老板一张图把控全局&#xff0c;运筹帷幄&#xff0c; 让员工每天的工作井然有序&#xff0c;让每件事的进展都有始有终、形成闭环。一起来看看吧&#xff01; 01企业内部任务管理的重要性 对于一家公司或一…

Java泛型的简单认识

泛型的认识 自定义泛型&#xff0c;定义了String类型&#xff0c;随后这个泛型就是String类型 于是他的方法都是字符串的类型 泛型接口 泛型方法 所有车可以进行比赛&#xff0c;定义了一个BMW和BENZ两个车类&#xff0c;都继承car&#xff0c;当使用泛型的 如果你顶一个狗对象…

Centos7 安装tomcat9

去官网下载 数据包 ps: wget https://dlcdn.apache.org/tomcat/tomcat-9/v9.0.78/bin/apache-tomcat-9.0.78.tar.gz检查Java环境 [tomcatlocalhost bin]$ java -version java version "1.8.0_121" Java(TM) SE Runtime Environment (build 1.8.0_121-b13) Java H…

品牌渠道治理的有效方法

什么是品牌渠道&#xff0c;即品牌的销售渠道&#xff0c;围绕销售渠道做的治理工作&#xff0c;根本上是对低价、窜货的治理&#xff0c;当渠道中存在低价问题&#xff0c;那不管是对经销商还是非授权店铺&#xff0c;都会有不好的影响&#xff0c;经销商会跟价&#xff0c;非…

派森编程软件python好学吗,派森语言python干什么的

大家好&#xff0c;小编来为大家解答以下问题&#xff0c;派森编程软件python有什么用&#xff0c;派森编程软件python好学吗&#xff0c;现在让我们一起来看看吧&#xff01; 1、python真的值得学吗&#xff1f; 不建议学python的原因&#xff1a; 1、语言性能差 对于C老手…

ACL访问控制列表(红茶三杯CCNA)

ACL的两大主要功能&#xff1a; 1. 流量控制 2. 匹配感兴趣流量ACL分为两类 Standard ACL-标准访问控制列表 只能根据源地址做过滤针对整个协议采取相关动作&#xff08;允许或禁止&#xff09; Extended ACL-扩展访问控制列表 能根据源、目的地址、端口号进行过滤能允许或拒…

实验四 汇编语言程序上机过程

五、实验步骤 1、用文字编辑工具(记事本或EDIT)将源程序输入,其扩展名为Hello.ASM。将源文件放到Dosbox挂载的目录下面。 图5-1为源码写注释并更改扩展名为Hello.ASM 2、用MASM对源文件进行汇编,产生Hello.OBJ文件和Hello.LST文件。若汇编时提示有错,用文字编辑工具修改源程…

YAPI接口自动鉴权功能部署详解

安装准备 以下操作&#xff0c;默认要求自己部署过yapi&#xff0c;最好是部署过yapi二次开发环境。 无论是选择在线安装或者是本地安装&#xff0c;都需要安装client工具。 1、yapi-cli&#xff1a;npm install yapi-cli –g&#xff0c; 2、安装后将文件夹nodejs/node_gl…

Nexpose v6.6.208 for Linux Windows - 漏洞扫描

Nexpose v6.6.208 for Linux & Windows - 漏洞扫描 Rapid7 Vulnerability Management, Release Jul 27, 2023 请访问原文链接&#xff1a;https://sysin.org/blog/nexpose-6/&#xff0c;查看最新版。原创作品&#xff0c;转载请保留出处。 作者主页&#xff1a;sysin.o…

10分钟带你实现一个Android自定义View:带动画的等级经验条

先展示一下静态效果图 介绍一下我们的实现流程&#xff1a; 首先整个经验条有一个圆角边框的背景打底&#xff1b;然后给经验条绘制一条轨道&#xff0c;让用户比较直观地看到总进度的长度&#xff1b;在轨道的上层绘制我们的渐变色经验条&#xff1b;在经验条的上层绘制等级…

vue新学习 02 vue命令v-model,数据代理(作用域和作用域链),事件,监听,渲染,计算属性(也就是把操作属性的语句放到vue实例中)

双向绑定用命令v-model&#xff1a; v-bind的命令是单项去绑定data中的相关属性&#xff0c;此时的data是真正的data&#xff0c;并没有用变量声明的方式去接收vue实例对象&#xff0c;也就是例如用const vm new Vue({})。而是直接就采用了new Vue&#xff08;{}&#xff09;这…

6、Kubernetes核心技术 - Pod

目录 一、概述 二、Pod机制 2.1、共享网络 2.2、共享存储 三、Pod资源清单 四、 Pod 的分类 五、Pod阶段 六、Pod 镜像拉取策略 ImagePullBackOff 七、Pod 资源限制 八、容器重启策略 一、概述 Pod 是可以在 Kubernetes 中创建和管理的、最小的可部署的计算单元。P…

oracle12c静默安装

目录 前言 安装配置步骤 关闭防火墙,禁止防火墙开机自启 关闭selinux yum安装必要安装包 内网环境下载依赖包的方式 创建用户和组 创建oinstall和dba组 创建oracle用户 设置oracle密码 查看创建结果 修改内核参数 使配置生效 修改用户及文件限制 改文件限制 修改用…