Promise(简介、基本使用、API、手写实现 Promise、async与await)(三)

简介: Promise(简介、基本使用、API、手写实现 Promise、async与await)(三)

5.2.4 测试

// 成功状态测试
    let p1 = new Promise( (resolve, reject)=>{
      resolve('OK')
    } )
    // p.then(val=>{
    //   console.log(val)
    // }, reason=>{
    //   console.log(reason)
    // })
    console.log(p1)
    // 失败状态测试
    let p2 = new Promise( (resolve, reject)=>{
      reject('error')
    } )
    console.log(p2)
    // 抛出异常测试
    let p3 = new Promise( (resolve, reject)=>{
      throw 'Error'
    } )
    console.log(p3)

会发现Promise的状态和结果没有进行更改

造成的原因:创建Promise对象时,传入的执行器函数为箭头函数,箭头函数没有自己的this,箭头函数的this会指向外层的this。

<body>
  <script>
  let p1 = new Promise( (resolve, reject)=>{
      resolve('OK')
    } )
  </script>
</body>

箭头函数的外层为页面,外层的this指向window,所以箭头函数的this指向window,因此需要想办法让传入执行器为箭头函数this指向Promise对象

5.2.5 保存实例对象的this

// 定义Promise的构造函数
function Promise(excutor) {
  // 定义 PromiseState PromiseResult 
  this.PromiseState = 'pending' // 默认状态 pending
  this.PromiseResult = null // 结果默认为空
  // 在构造 Promise 对象时将 Promise 对象的 this 保存下来
  // 修改时修改 self 上的属性即可,self 指向本对象的this
  // self 就是本对象
  const self = this
  // 定义resolve函数
  function resolve(data) {
    if (self.PromiseState !== 'pending') return // 状态已经修改直接退出
    self.PromiseState = 'fulfilled' // 修改 Promise的状态 PromiseState
    self.PromiseResult = data       // 修改 Promise的结果 PromiseResult
  }
  // 定义reject函数
  function reject(data) {
    if (self.PromiseState !== 'pending') return // 状态已经修改直接退出
    self.PromiseState = 'rejected'  // 修改 Promise的状态 PromiseState
    self.PromiseResult = data       // 修改 Promise的结果 PromiseResult
  }
  // 捕获 excutor 中的异常并进行处理
  try {
    // 调用传参传过来的执行器函数  同步调用
    excutor(resolve, reject)
  } catch (e) {
    // 处理异常,改变Promise状态为失败,改变结果
    reject(e)
  }
}

5.2.6 再测试

<body>
  <script>
    // 成功状态测试
    let p1 = new Promise( (resolve, reject)=>{
      resolve('OK')
    } )
    // p.then(val=>{
    //   console.log(val)
    // }, reason=>{
    //   console.log(reason)
    // })
    console.log(p1)
    // 失败状态测试
    let p2 = new Promise( (resolve, reject)=>{
      reject('error')
    } )
    console.log(p2)
    // 抛出异常测试
    let p3 = new Promise( (resolve, reject)=>{
      throw 'Error'
    } )
    console.log(p3)
  </script>
</body>

后面补的执行图

5.3 promise.then()的实现

5.3.1 then() 实现

// 定义 Promise 的 then 方法
// 定义在 prototype 上
// then 方法中有两个函数类型的参数 成功状态的回调函数和失败状态的回调函数
Promise.prototype.then = function (onResolved, onRejected) {
  // 根据 Promise 的状态 PromiseState 判断要调用哪个回调函数
  // 状态为成功
  if (this.PromiseState === 'fulfilled' || this.PromiseState === 'resolved') {
    onResolved( this.PromiseResult )
  }
  // 状态为失败
  if (this.PromiseState === 'rejected') {
    onRejected( this.PromiseResult )
  }
}

then 方法不为箭头函数,为普通函数,有自己的this,指向调用该方法的实例对象

测试:

<body>
  <script>
    // 成功状态测试
    console.log('===== 成功状态测试 =====')
    let p1 = new Promise( (resolve, reject)=>{
      resolve('OK')
    } )
    p1.then(val=>{
      console.log(val)
    }, reason=>{
      console.log(reason)
    })
    console.log(p1)
    // 失败状态测试
    console.log('===== 失败状态测试 =====')
    let p2 = new Promise( (resolve, reject)=>{
      reject('error')
    } )
    p2.then(val=>{
      console.log(val)
    }, reason=>{
      console.log(reason)
    })
    console.log(p2)
    // 抛出异常测试
    console.log('===== 抛出异常测试 =====')
    let p3 = new Promise( (resolve, reject)=>{
      throw 'Error'
    } )
    p3.then(val=>{
      console.log(val)
    }, reason=>{
      console.log(reason)
    })
    console.log(p3)
  </script>
</body>

5.3.2 then() 异步回调执行实现

执行器函数中改为异步任务,会发现没有任何输出

let p1 = new Promise( (resolve, reject)=>{
      setTimeout(()=>{
        resolve('OK')
      }, 500)
    } )
    p1.then(val=>{
      console.log(val)
    }, reason=>{
      console.log(reason)
    })

原因:

所以 then() 中的回调函数并未执行,不会有任何的输出。then() 中缺少对于pending状态时候的判断,需要增加对于pending状态的判断。

// 定义 Promise 的 then 方法
// 定义在 prototype 上
// then 方法中有两个函数类型的参数 成功状态的回调函数和失败状态的回调函数
Promise.prototype.then = function (onResolved, onRejected) {
  // 根据 Promise 的状态 PromiseState 判断要调用哪个回调函数
  // 状态为成功
  if (this.PromiseState === 'fulfilled' || this.PromiseState === 'resolved') {
    onResolved(this.PromiseResult)
    console.log(this)
  }
  // 状态为失败
  if (this.PromiseState === 'rejected') {
    onRejected(this.PromiseResult)
    console.log(this)
  }
  // 状态为 pending
  if (this.PromiseState === 'pending') {
    // then() 中回调函数执行的时机为 Promise 状态改变之后
    // 所以回调函数应该在状态改变之后进行调用
    // 因此回调函数在resolve或reject函数中转台改变之后进行调用
    // 但是resolve和reject与then为不同的函数,不同的函数作用域
    // 需要将回调函数进行保存后才可以在resolve或reject中进行调用
    // 在构造函数中声明一个对象来保存回调函数
    // 在then()使用其来保存回调函数
    this.callback = {
      onResolved: onResolved,
      onRejected: onRejected
    }
    // 简写
    // this.callback = {
    //   onResolved,
    //   onRejected
    // }
  }
}
// 在构造函数中声明一个回调函数对象用于保存回调函数,
  // 以便在resolve或reject中执行
  this.callback = {}
// 定义Promise的构造函数
function Promise(excutor) {
  // 定义 PromiseState PromiseResult 
  this.PromiseState = 'pending' // 默认状态 pending
  this.PromiseResult = null // 结果默认为空
  const self = this // 将 Promise 对象的 this 保存下来
  // 在构造函数中声明一个回调函数对象用于保存回调函数,
  // 以便在resolve或reject中执行
  this.callback = {}
  // 定义resolve函数
  function resolve(data) {
    if (self.PromiseState !== 'pending') return // 状态已经修改直接退出
    self.PromiseState = 'fulfilled' // 修改 Promise的状态 PromiseState
    self.PromiseResult = data       // 修改 Promise的结果 PromiseResult
    // 状态改变之后有成功的回调函数,调用成功的回调函数
    if (self.callback.onResolved) {
      self.callback.onResolved(self.PromiseResult);
    }
  }
  // 定义reject函数
  function reject(data) {
    if (self.PromiseState !== 'pending') return // 状态已经修改直接退出
    self.PromiseState = 'rejected'  // 修改 Promise的状态 PromiseState
    self.PromiseResult = data       // 修改 Promise的结果 PromiseResult
    // 状态改变之后有失败的回调函数,调用失败的回调函数
    if (self.callback.onResolved) {
      self.callback.onResolved(self.PromiseResult);
    }
  }
  // 捕获 excutor 中的异常并进行处理
  try {
    // 调用传参传过来的执行器函数  同步调用
    excutor(resolve, reject)
  } catch (e) {
    // 处理异常,改变Promise状态为失败,改变结果
    reject(e)
  }

测试:

let p1 = new Promise( (resolve, reject)=>{
      setTimeout(()=>{
        resolve('OK')
      }, 500)
    } )
    p1.then(val=>{
      console.log(val)
    }, reason=>{
      console.log(reason)
    })
    let p2 = new Promise( (resolve, reject)=>{
      setTimeout(()=>{
        reject('err')
      }, 500)
    } )
    p2.then(val=>{
      console.log(val)
    }, reason=>{
      console.log(reason)
    })

5.3.3 then() 指定多个回调函数

给p1多指定一个回调函数,会发现只有最后指定的回调函数被执行(原因:后面指定的回调函数覆盖了前面的回调函数)

let p1 = new Promise( (resolve, reject)=>{
      setTimeout(()=>{
        resolve('OK')
      }, 500)
    } )
    p1.then(val=>{
      console.log(val)
    }, reason=>{
      console.log(reason)
    })
    p1.then(val=>{
      console.log(val+'!!!!!!!')
    }, reason=>{
      console.log(reason+'!!!!!!!')
    })

要执行多个回调函数,那么就需要将每个回调函数进行保存,因此改用数组保存回调函数。

this.callback = [] // 声明一个数组用于保存回调函数
• 1
// 状态为 pending
  if (this.PromiseState === 'pending') {
    this.callback.push({
      onResolved: onResolved,
      onRejected: onRejected
    })
  }
// 定义resolve函数
  function resolve(data) {
    if (self.PromiseState !== 'pending') return // 状态已经修改直接退出
    self.PromiseState = 'fulfilled' // 修改 Promise的状态 PromiseState
    self.PromiseResult = data       // 修改 Promise的结果 PromiseResult
    self.callback.forEach(item=>{ // 调用所有的回调函数
      item.onResolved(self.PromiseResult)
    });
  }
  // 定义reject函数
  function reject(data) {
    if (self.PromiseState !== 'pending') return // 状态已经修改直接退出
    self.PromiseState = 'rejected'  // 修改 Promise的状态 PromiseState
    self.PromiseResult = data       // 修改 Promise的结果 PromiseResult
    self.callback.forEach(item=>{ // 调用所有的回调函数
      item.onRejected(self.PromiseResult)
    });
  }

完整代码:

// 定义Promise的构造函数
function Promise(excutor) {
  // 定义 PromiseState PromiseResult 
  this.PromiseState = 'pending' // 默认状态 pending
  this.PromiseResult = null // 结果默认为空
  const self = this // 将 Promise 对象的 this 保存下来
  this.callback = [] // 声明一个数组用于保存回调函数
  // 定义resolve函数
  function resolve(data) {
    if (self.PromiseState !== 'pending') return // 状态已经修改直接退出
    self.PromiseState = 'fulfilled' // 修改 Promise的状态 PromiseState
    self.PromiseResult = data       // 修改 Promise的结果 PromiseResult
    self.callback.forEach(item=>{ // 调用所有的回调函数
      item.onResolved(self.PromiseResult)
    });
  }
  // 定义reject函数
  function reject(data) {
    if (self.PromiseState !== 'pending') return // 状态已经修改直接退出
    self.PromiseState = 'rejected'  // 修改 Promise的状态 PromiseState
    self.PromiseResult = data       // 修改 Promise的结果 PromiseResult
    self.callback.forEach(item=>{ // 调用所有的回调函数
      item.onRejected(self.PromiseResult)
    });
  }
  // 捕获 excutor 中的异常并进行处理
  try {
    // 调用传参传过来的执行器函数  同步调用
    excutor(resolve, reject)
  } catch (e) {
    // 处理异常,改变Promise状态为失败,改变结果
    reject(e)
  }
}
// 定义 Promise 的 then 方法
// 定义在 prototype 上
// then 方法中有两个函数类型的参数 成功状态的回调函数和失败状态的回调函数
Promise.prototype.then = function (onResolved, onRejected) {
  // 根据 Promise 的状态 PromiseState 判断要调用哪个回调函数
  // 状态为成功
  if (this.PromiseState === 'fulfilled' || this.PromiseState === 'resolved') {
    onResolved(this.PromiseResult)
    console.log(this)
  }
  // 状态为失败
  if (this.PromiseState === 'rejected') {
    onRejected(this.PromiseResult)
    console.log(this)
  }
  // 状态为 pending
  if (this.PromiseState === 'pending') {
    this.callback.push({
      onResolved: onResolved,
      onRejected: onRejected
    })

测试:

let p1 = new Promise( (resolve, reject)=>{
      setTimeout(()=>{
        resolve('OK')
      }, 500)
    } )
    p1.then(val=>{
      console.log(val)
    }, reason=>{
      console.log(reason)
    })
    p1.then(val=>{
      console.log(val+'!!!!!!!')
    }, reason=>{
      console.log(reason+'!!!!!!!')
    })
    let p2 = new Promise( (resolve, reject)=>{
      setTimeout(()=>{
        reject('err')
      }, 500)
    } )
    p2.then(val=>{
      console.log(val)
    }, reason=>{
      console.log(reason)
    })


相关文章
|
2月前
|
前端开发 JavaScript 开发者
Async 和 Await 是基于 Promise 实现
【10月更文挑战第30天】Async和Await是基于Promise实现的语法糖,它们通过简洁的语法形式,借助Promise的异步处理机制,为JavaScript开发者提供了一种更优雅、更易于理解和维护的异步编程方式。
33 1
|
2月前
|
前端开发
如何使用async/await解决Promise的缺点?
总的来说,`async/await` 是对 Promise 的一种很好的补充和扩展,它为我们提供了更高效、更易读、更易维护的异步编程方式。通过合理地运用 `async/await`,我们可以更好地解决 Promise 的一些缺点,提升异步代码的质量和开发效率。
36 5
|
2月前
|
前端开发 JavaScript
async/await和Promise在性能上有什么区别?
性能优化是一个综合性的工作,除了考虑异步模式的选择外,还需要关注代码的优化、资源的合理利用等方面。
40 4
|
2月前
|
JSON 前端开发 JavaScript
浅谈JavaScript中的Promise、Async和Await
【10月更文挑战第30天】Promise、Async和Await是JavaScript中强大的异步编程工具,它们各自具有独特的优势和适用场景,开发者可以根据具体的项目需求和代码风格选择合适的方式来处理异步操作,从而编写出更加高效、可读和易于维护的JavaScript代码。
35 1
|
3月前
|
前端开发 JavaScript
setTimeout、Promise、Async/Await 的区别
`setTimeout` 是用于延迟执行函数的简单方法;`Promise` 表示异步操作的最终完成或失败;`Async/Await` 是基于 Promise 的语法糖,使异步代码更易读和维护。三者都用于处理异步操作,但使用场景和语法有所不同。
|
3月前
|
前端开发 JavaScript 开发者
JavaScript 中的异步编程:深入了解 Promise 和 async/await
【10月更文挑战第8天】JavaScript 中的异步编程:深入了解 Promise 和 async/await
|
3月前
|
前端开发 JavaScript UED
深入了解JavaScript异步编程:回调、Promise与async/await
【10月更文挑战第11天】深入了解JavaScript异步编程:回调、Promise与async/await
26 0
|
API 计算机视觉
MLDL之API:关于各国内外大平台API简介、使用方法之详细攻略
MLDL之API:关于各国内外大平台API简介、使用方法之详细攻略
|
20天前
|
人工智能 自然语言处理 API
Multimodal Live API:谷歌推出新的 AI 接口,支持多模态交互和低延迟实时互动
谷歌推出的Multimodal Live API是一个支持多模态交互、低延迟实时互动的AI接口,能够处理文本、音频和视频输入,提供自然流畅的对话体验,适用于多种应用场景。
67 3
Multimodal Live API:谷歌推出新的 AI 接口,支持多模态交互和低延迟实时互动
|
7天前
|
JSON 安全 API
淘宝商品详情API接口(item get pro接口概述)
淘宝商品详情API接口旨在帮助开发者获取淘宝商品的详细信息,包括商品标题、描述、价格、库存、销量、评价等。这些信息对于电商企业而言具有极高的价值,可用于商品信息展示、市场分析、价格比较等多种应用场景。