zoukankan      html  css  js  c++  java
  • 手写简易版Promise

    实现一个简易版 Promise

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

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

    const PENDING = 'pending'
    const RESOLVED = 'resolved'
    const REJECTED = 'rejected'
    
    function MyPromise(fn) {
      const that = this
      that.state = PENDING
      that.value = null
      that.resolvedCallbacks = []
      that.rejectedCallbacks = []
      // 待完善 resolve 和 reject 函数
      // 待完善执行 fn 函数
    }
    • 首先我们创建了三个常量用于表示状态,对于经常使用的一些值都应该通过常量来管理,便于开发及后期维护
    • 在函数体内部首先创建了常量 that,因为代码可能会异步执行,用于获取正确的 this 对象
    • 一开始 Promise 的状态应该是 pending
    • value 变量用于保存 resolve 或者 reject 中传入的值
    • resolvedCallbacks 和 rejectedCallbacks 用于保存 then 中的回调,因为当执行完 Promise 时状态可能还是等待中,这时候应该把 then 中的回调保存起来用于状态改变时使用

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

    function resolve(value) {
      if (that.state === PENDING) {
        that.state = RESOLVED
        that.value = value
        that.resolvedCallbacks.map(cb => cb(that.value))
      }
    }
    
    function reject(value) {
      if (that.state === PENDING) {
        that.state = REJECTED
        that.value = value
        that.rejectedCallbacks.map(cb => cb(that.value))
      }
    }

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

    • 首先两个函数都得判断当前状态是否为等待中,因为规范规定只有等待态才可以改变状态
    • 将当前状态更改为对应状态,并且将传入的值赋值给 value
    • 遍历回调数组并执行

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

    try {
      fn(resolve, reject)
    } catch (e) {
      reject(e)
    }
    • 实现很简单,执行传入的参数并且将之前两个函数当做参数传进去
    • 要注意的是,可能执行函数过程中会遇到错误,需要捕获错误并且执行 reject 函数

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

    MyPromise.prototype.then = function(onFulfilled, onRejected) {
      const that = this
      onFulfilled = typeof onFulfilled === 'function' ? onFulfilled : v => v
      onRejected =
        typeof onRejected === 'function'
          ? onRejected
          : r => {
              throw r
            }
      if (that.state === PENDING) {
        that.resolvedCallbacks.push(onFulfilled)
        that.rejectedCallbacks.push(onRejected)
      }
      if (that.state === RESOLVED) {
        onFulfilled(that.value)
      }
      if (that.state === REJECTED) {
        onRejected(that.value)
      }
    }
    • 首先判断两个参数是否为函数类型,因为这两个参数是可选参数

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

      // 该代码目前在简单版中会报错
      // 只是作为一个透传的例子
      Promise.resolve(4).then().then((value) => console.log(value))
    • 接下来就是一系列判断状态的逻辑,当状态不是等待态时,就去执行相对应的函数。如果状态是等待态的话,就往回调函数中 push 函数,比如如下代码就会进入等待态的逻辑

      new MyPromise((resolve, reject) => {
        setTimeout(() => {
          resolve(1)
        }, 0)
      }).then(value => {
        console.log(value)
      })

    以上就是简单版 Promise 实现

  • 相关阅读:
    C# 上传下载文件
    ASP.Net
    C#计算程序执行时间
    关于vs连接access
    CSS
    C# 获取命名空间,类,方法名
    C# Split
    C#方法参数:params,ref,out,可选参数,命名参数
    java学习手记:JDK的安装配置
    Mysql/SQLServer数据类型与java基本数据类型的对应
  • 原文地址:https://www.cnblogs.com/keyng/p/12942779.html
Copyright © 2011-2022 走看看