在完成符合 Promise/A+ 规范的代码之前,我们可以先来实现一个简易版 Promise,因为在面试中,如果你能实现出一个简易版的 Promise 基本可以过关了。

那么我们先来搭建构建函数的大体框架

  1. const PENDING = 'pending'
  2. const RESOLVED = 'resolved'
  3. const REJECTED = 'rejected'
  4. function MyPromise(fn) {
  5. const that = this
  6. that.state = PENDING
  7. that.value = null
  8. that.resolvedCallbacks = []
  9. that.rejectedCallbacks = []
  10. // 待完善 resolve 和 reject 函数
  11. // 待完善执行 fn 函数
  12. }

首先我们创建了三个常量用于表示状态,对于经常使用的一些值都应该通过常量来管理,便于开发及后期维护

在函数体内部首先创建了常量 that,因为代码可能会异步执行,用于获取正确的 this 对象

一开始 Promise 的状态应该是 pending

value 变量用于保存 resolve 或者 reject 中传入的值

resolvedCallbacks 和 rejectedCallbacks 用于保存 then 中的回调,因为当执行完 Promise 时状态可能还是等待中,这时候应该把 then 中的回调保存起来用于状态改变时使用

接下来我们来完善 resolve 和 reject 函数,添加在 MyPromise 函数体内部

  1. function resolve(value) {
  2. if (that.state === PENDING) {
  3. that.state = RESOLVED
  4. that.value = value
  5. that.resolvedCallbacks.map(cb => cb(that.value))
  6. }
  7. }
  8. function reject(value) {
  9. if (that.state === PENDING) {
  10. that.state = REJECTED
  11. that.value = value
  12. that.rejectedCallbacks.map(cb => cb(that.value))
  13. }
  14. }

这两个函数代码类似,就一起解析了

首先两个函数都得判断当前状态是否为等待中,因为规范规定只有等待态才可以改变状态

将当前状态更改为对应状态,并且将传入的值赋值给 value

遍历回调数组并执行

完成以上两个函数以后,我们就该实现如何执行 Promise 中传入的函数了

  1. try {
  2. fn(resolve, reject)
  3. } catch (e) {
  4. reject(e)
  5. }

实现很简单,执行传入的参数并且将之前两个函数当做参数传进去

要注意的是,可能执行函数过程中会遇到错误,需要捕获错误并且执行 reject 函数

最后我们来实现较为复杂的 then 函数

  1. MyPromise.prototype.then = function(onFulfilled, onRejected) {
  2. const that = this
  3. onFulfilled = typeof onFulfilled === 'function' ? onFulfilled : v => v
  4. onRejected =
  5. typeof onRejected === 'function'
  6. ? onRejected
  7. : r => {
  8. throw r
  9. }
  10. if (that.state === PENDING) {
  11. that.resolvedCallbacks.push(onFulfilled)
  12. that.rejectedCallbacks.push(onRejected)
  13. }
  14. if (that.state === RESOLVED) {
  15. onFulfilled(that.value)
  16. }
  17. if (that.state === REJECTED) {
  18. onRejected(that.value)
  19. }
  20. }

首先判断两个参数是否为函数类型,因为这两个参数是可选参数

当参数不是函数类型时,需要创建一个函数赋值给对应的参数,同时也实现了透传,比如如下代码

// 该代码目前在简单版中会报错

// 只是作为一个透传的例子

Promise.resolve(4).then().then((value) => console.log(value))

接下来就是一系列判断状态的逻辑,当状态不是等待态时,就去执行相对应的函数。如果状态是等待态的话,就往回调函数中 push 函数,比如如下代码就会进入等待态的逻辑

  1. new MyPromise((resolve, reject) => {
  2. setTimeout(() => {
  3. resolve(1)
  4. }, 0)
  5. }).then(value => {
  6. console.log(value)
  7. })

以上就是简单版 Promise 实现,接下来一小节是实现完整版 Promise 的解析,相信看完完整版的你,一定会对于 Promise 的理解更上一层楼。

实现一个符合 Promise/A+ 规范的 Promise

这小节代码需要大家配合规范阅读,因为大部分代码都是根据规范去实现的。

我们先来改造一下 resolve 和 reject 函数

  1. function resolve(value) {
  2. if (value instanceof MyPromise) {
  3. return value.then(resolve, reject)
  4. }
  5. setTimeout(() => {
  6. if (that.state === PENDING) {
  7. that.state = RESOLVED
  8. that.value = value
  9. that.resolvedCallbacks.map(cb => cb(that.value))
  10. }
  11. }, 0)
  12. }
  13. function reject(value) {
  14. setTimeout(() => {
  15. if (that.state === PENDING) {
  16. that.state = REJECTED
  17. that.value = value
  18. that.rejectedCallbacks.map(cb => cb(that.value))
  19. }
  20. }, 0)
  21. }

对于 resolve 函数来说,首先需要判断传入的值是否为 Promise 类型

为了保证函数执行顺序,需要将两个函数体代码使用 setTimeout 包裹起来

接下来继续改造 then 函数中的代码,首先我们需要新增一个变量 promise2,因为每个 then 函数都需要返回一个新的 Promise 对象,该变量用于保存新的返回对象,然后我们先来改造判断等待态的逻辑

  1. if (that.state === PENDING) {
  2. return (promise2 = new MyPromise((resolve, reject) => {
  3. that.resolvedCallbacks.push(() => {
  4. try {
  5. const x = onFulfilled(that.value)
  6. resolutionProcedure(promise2, x, resolve, reject)
  7. } catch (r) {
  8. reject(r)
  9. }
  10. })
  11. that.rejectedCallbacks.push(() => {
  12. try {
  13. const x = onRejected(that.value)
  14. resolutionProcedure(promise2, x, resolve, reject)
  15. } catch (r) {
  16. reject(r)
  17. }
  18. })
  19. }))
  20. }

首先我们返回了一个新的 Promise 对象,并在 Promise 中传入了一个函数

函数的基本逻辑还是和之前一样,往回调数组中 push 函数

同样,在执行函数的过程中可能会遇到错误,所以使用了 try...catch 包裹

规范规定,执行 onFulfilled 或者 onRejected 函数时会返回一个 x,并且执行 Promise 解决过程,这是为了不同的 Promise 都可以兼容使用,比如 JQuery 的 Promise 能兼容 ES6 的 Promise

接下来我们改造判断执行态的逻辑

  1. if (that.state === RESOLVED) {
  2. return (promise2 = new MyPromise((resolve, reject) => {
  3. setTimeout(() => {
  4. try {
  5. const x = onFulfilled(that.value)
  6. resolutionProcedure(promise2, x, resolve, reject)
  7. } catch (reason) {
  8. reject(reason)
  9. }
  10. })
  11. }))
  12. }

其实大家可以发现这段代码和判断等待态的逻辑基本一致,无非是传入的函数的函数体需要异步执行,这也是规范规定的

对于判断拒绝态的逻辑这里就不一一赘述了,留给大家自己完成这个作业

最后,当然也是最难的一部分,也就是实现兼容多种 Promise 的 resolutionProcedure 函数

  1. function resolutionProcedure(promise2, x, resolve, reject) {
  2. if (promise2 === x) {
  3. return reject(new TypeError('Error'))
  4. }
  5. }

首先规范规定了 x 不能与 promise2 相等,这样会发生循环引用的问题,比如如下代码

  1. let p = new MyPromise((resolve, reject) => {
  2. resolve(1)
  3. })
  4. let p1 = p.then(value => {
  5. return p1
  6. })

然后需要判断 x 的类型

  1. if (x instanceof MyPromise) {
  2. x.then(function(value) {
  3. resolutionProcedure(promise2, value, resolve, reject)
  4. }, reject)
  5. }

这里的代码是完全按照规范实现的。如果 x 为 Promise 的话,需要判断以下几个情况:

如果 x 处于等待态,Promise 需保持为等待态直至 x 被执行或拒绝

如果 x 处于其他状态,则用相同的值处理 Promise

当然以上这些是规范需要我们判断的情况,实际上我们不判断状态也是可行的。

接下来我们继续按照规范来实现剩余的代码

  1. let called = false
  2. if (x !== null && (typeof x === 'object' || typeof x === 'function')) {
  3. try {
  4. let then = x.then
  5. if (typeof then === 'function') {
  6. then.call(
  7. x,
  8. y => {
  9. if (called) return
  10. called = true
  11. resolutionProcedure(promise2, y, resolve, reject)
  12. },
  13. e => {
  14. if (called) return
  15. called = true
  16. reject(e)
  17. }
  18. )
  19. } else {
  20. resolve(x)
  21. }
  22. } catch (e) {
  23. if (called) return
  24. called = true
  25. reject(e)
  26. }
  27. } else {
  28. resolve(x)
  29. }

首先创建一个变量 called 用于判断是否已经调用过函数

然后判断 x 是否为对象或者函数,如果都不是的话,将 x 传入 resolve 中

如果 x 是对象或者函数的话,先把 x.then 赋值给 then,然后判断 then 的类型,如果不是函数类型的话,就将 x 传入 resolve 中

如果 then 是函数类型的话,就将 x 作为函数的作用域 this 调用之,并且传递两个回调函数作为参数,第一个参数叫做 resolvePromise ,第二个参数叫做 rejectPromise,两个回调函数都需要判断是否已经执行过函数,然后进行相应的逻辑

以上代码在执行的过程中如果抛错了,将错误传入 reject 函数中

手写 Promise的更多相关文章

  1. 手写Promise A+ 规范

    基于ES6语法手写promise A+ 规范,源码实现 class Promise { constructor(excutorCallBack) { this.status = 'pending'; ...

  2. 手写promise

    写在前面: 在目前的前端分开中,我们对于异步方法的使用越来越频繁,那么如果处理异步方法的返回结果,如果优雅的进行异步处理对于一个合格的前端开发者而言就显得尤为重要,其中在面试中被问道最多的就是对Pro ...

  3. 手写Promise看着一篇就足够了

    目录 概要 博客思路 API的特性与手写源码 构造函数 then catch Promise.resolved Promise.rejected Promise.all Promise.race 概要 ...

  4. 手写Promise中then方法返回的结果或者规律

    1. Promise中then()方法返回来的结果或者规律 我们知道 promise 的 then 方法返回来的结果值[result]是由: 它指定的回调函数的结果决定的 2.比如说下面这一段代码 l ...

  5. 前端面试题之手写promise

    前端面试题之Promise问题 前言 在我们日常开发中会遇到很多异步的情况,比如涉及到 网络请求(ajax,axios等),定时器这些,对于这些异步操作我们如果需要拿到他们操作后的结果,就需要使用到回 ...

  6. [转]史上最最最详细的手写Promise教程

    我们工作中免不了运用promise用来解决异步回调问题.平时用的很多库或者插件都运用了promise 例如axios.fetch等等.但是你知道promise是咋写出来的呢? 别怕-这里有本promi ...

  7. 手写Promise简易版

    话不多说,直接上代码 通过ES5的模块化封装,向外暴露一个属性 (function(window){ const PENDING = 'pending'; const RESOLVED = 'fulf ...

  8. 史上最简单的手写Promise,仅17行代码即可实现Promise链式调用

    Promise的使用相比大家已经孰能生巧了,我这里就不赘述了 先说说我写的Promise的问题吧,无法实现宏任务和微任务里的正确执行(也就是在Promise里面写setTimeout,setInter ...

  9. js手写'Promise'

    /* * pending:初始化成功 * fulfilled:成功 * rejected:失败 * */ function Promise(executor) {// 执行器 this.status ...

  10. js 手写 Promise

    /* * pending:初始化成功 * fulfilled:成功 * rejected:失败 * */ function Promise(cback){ this.status = 'pending ...

随机推荐

  1. 常用linux 命令

    ls -lt 时间倒序 ls -ltr 时间正序 ls -lS 大小倒序 ls -li 显示inode ----------------- cat smb_quicktest.py| grep -E ...

  2. 17.python内置函数2

    python内置函数1:https://www.cnblogs.com/raitorei/p/11813694.html # max,min高级玩法 # l=[1,3,100,-1,2] # prin ...

  3. b方式操作文件

    f=open('test11.py','rb',encoding='utf-8') #b的方式不能指定编码 f=open('test11.py','rb') #b的方式不能指定编码 data=f.re ...

  4. __FILE__,__LINE__,__DATE__,__TIME__,__FUNCTION__的使用

    C/C++ 有5个常用的预定义宏,可以当作变量直接使用 __FILE__,__FUNCTION __,__LINE__,__DATE__,__TIME__. 注意是两个下划线. 其含义如下: __FI ...

  5. 洛谷$P3308\ [SDOI2014]LIS$ 网络流

    正解:网络流 解题报告: 传送门$QwQ$ 恩先不考虑关于那个附加属性的限制,考虑这题怎么做? 首先这题从名字开始就让人忍不住联想起网络流24题里的那个最长不下降子序列?于是同样考虑预处理一个$f$呗 ...

  6. Java日志体系居然这么复杂?——架构篇

    本文是一个系列,欢迎关注 日志到底是何方神圣?为什么要使用日志框架? 想必大家都有过使用System.out来进行输出调试,开发开发环境下这样做当然很方便,但是线上这样做就有麻烦了: 系统一直运行,输 ...

  7. UML类图基础

    UML( Unified Modeling Language) 统一建模语言, 它是一个支持模型化和软件系统开发的图形化语言,为软件开发的所有阶段提供模型化和可视化支持,包括由需求分析到规格,到构造和 ...

  8. 洛谷P1082 同余方程 题解

    题目链接:https://www.luogu.com.cn/problem/P1082 题目大意: 求关于 \(x\) 的同余方程 ax≡1(mod b) 的最小正整数解. 告诉你 \(a,b\) 求 ...

  9. [AI开发]零代码分析视频结构化类应用结构设计

    视频结构化类应用涉及到的技术栈比较多,而且每种技术入门门槛都较高,比如视频接入存储.编解码.深度学习推理.rtmp流媒体等等.每个环节的水都非常深,单独拿出来可以写好几篇文章,如果没有个几年经验基本很 ...

  10. 现代主流框架路由原理 hash、history的底层原理

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...