文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

使用 Promise 时的五个常见错误,你占了几个

2024-12-02 16:17

关注

Promise 提供了一种优雅的方法来处理 JS 中的异步操作。这也是避免“回调地狱”的解决方案。然而,并没有多少开发人员了解其中的内容。因此,许多人在实践中往往会犯错误。

在本文中,介绍一下使用 promise 时的五个常见错误,希望大家能够避免这些错误。

1.避免 Promise 地狱

通常,Promise是用来避免回调地狱。但滥用它们也会导致 Promise是地狱。

  1. userLogin('user').then(function(user){ 
  2.     getArticle(user).then(function(articles){ 
  3.         showArticle(articles).then(function(){ 
  4.             //Your code goes here... 
  5.         }); 
  6.     }); 
  7. }); 

 在上面的例子中,我们对 userLogin、getararticle 和 showararticle 嵌套了三个promise。这样复杂性将按代码行比例增长,它可能变得不可读。

为了避免这种情况,我们需要解除代码的嵌套,从第一个 then 中返回 getArticle,然后在第二个 then 中处理它。

  1. userLogin('user'
  2.   .then(getArticle) 
  3.   .then(showArticle) 
  4.   .then(function(){ 
  5.        //Your code goes here... 
  6. }); 

2. 在 Promise 中使用try/catch块

通常情况下,我们使用 try/catch 块来处理错误。然而,不建议在 Promise 对象中使用try/catch 。

这是因为如果有任何错误,Promise对象会在 catch 内自动处理。

  1. ew Promise((resolve, reject) => { 
  2.   try { 
  3.     const data = doThis(); 
  4.     // do something 
  5.     resolve(); 
  6.   } catch (e) { 
  7.     reject(e); 
  8.   } 
  9. }) 
  10.   .then(data => console.log(data)) 
  11.   .catch(error => console.log(error)); 

在上面的例子中,我们在Promise 内使用了 try/catch 块。

但是,Promise本身会在其作用域内捕捉所有的错误(甚至是打字错误),而不需要 try/catch块。它确保在执行过程中抛出的所有异常都被获取并转换为被拒绝的 Promise。

  1. new Promise((resolve, reject) => { 
  2.   const data = doThis(); 
  3.   // do something 
  4.   resolve() 
  5. }) 
  6.   .then(data => console.log(data)) 
  7.   .catch(error => console.log(error)); 

 **注意:**在 Promise 块中使用 .catch() 块是至关重要的。否则,你的测试案例可能会失败,而且应用程序在生产阶段可能会崩溃。

3. 在 Promise 块内使用异步函数

Async/Await 是一种更高级的语法,用于处理同步代码中的多个Promise。当我们在一个函数声明前使用 async 关键字时,它会返回一个 Promise,我们可以使用 await 关键字来停止代码,直到我们正在等待的Promise解决或拒绝。

但是,当你把一个 Async 函数放在一个 Promise 块里面时,会有一些副作用。

假设我们想在Promise 块中做一个异步操作,所以使用了 async 关键字,但,不巧的是我们的代码抛出了一个错误。

这样,即使使用 catch() 块或在 try/catch 块内等待你的Promise,我们也不能立即处理这个错误。请看下面的例子。

  1. // 此代码无法处理错误 
  2. new Promise(async () => { 
  3.   throw new Error('message'); 
  4. }).catch(e => console.log(e.message)); 
  5.  
  6. (async () => { 
  7.   try { 
  8.     await new Promise(async () => { 
  9.       throw new Error('message'); 
  10.     }); 
  11.   } catch (e) { 
  12.     console.log(e.message); 
  13.   } 
  14. })(); 

 当我在Promise块内遇到 async 函数时,我试图将 async 逻辑保持在 Promise 块之外,以保持其同步性。10次中有9次都能成功。

然而,在某些情况下,可能需要一个 async 函数。在这种情况下,也别无选择,只能用try/catch 块来手动管理。

  1. new Promise(async (resolve, reject) => { 
  2.   try { 
  3.     throw new Error('message'); 
  4.   } catch (error) { 
  5.     reject(error); 
  6.   } 
  7. }).catch(e => console.log(e.message)); 
  8.  
  9.  
  10. //using async/await 
  11. (async () => { 
  12.   try { 
  13.     await new Promise(async (resolve, reject) => { 
  14.       try { 
  15.         throw new Error('message'); 
  16.       } catch (error) { 
  17.         reject(error); 
  18.       } 
  19.     }); 
  20.   } catch (e) { 
  21.     console.log(e.message); 
  22.   } 
  23. })(); 

4.在创建 Promise 后立即执行 Promise 块

至于下面的代码片断,如果我们把代码片断放在调用HTTP请求的地方,它就会被立即执行。

  1. const myPromise = new Promise(resolve => { 
  2.   // code to make HTTP request 
  3.   resolve(result); 
  4. }); 

原因是这段代码被包裹在一个Promise构造函数中。然而,有些人可能会认为只有在执行myPromise 的then方法之后才被触发。

然而,真相并非如此。相反,当一个Promise被创建时,回调被立即执行。

这意味着在建立 myPromise 之后到达下面一行时,HTTP请求很可能已经在运行,或者至少处于调度状态。

Promises 总是急于执行过程。

但是,如果希望以后再执行 Promises,应该怎么做?如果现在不想发出HTTP请求怎么办?是否有什么神奇的机制内置于 Promises 中,使我们能够做到这一点?

答案就是使用函数。函数是一种耗时的机制。只有当开发者明确地用 () 来调用它们时,它们才会执行。简单地定义一个函数还不能让我们得到什么。所以,让 Promise 变得懒惰的最有效方法是将其包裹在一个函数中!

  1. const createMyPromise = () => new Promise(resolve => { 
  2.   // HTTP request 
  3.   resolve(result); 
  4. }); 

对于HTTP请求,Promise 构造函数和回调函数只有在函数被执行时才会被调用。所以现在我们有一个懒惰的Promise,只有在我们需要的时候才会执行。

5. 不一定使用 Promise.all() 方法

如果你已经工作多年,应该已经知道我在说什么了。如果有许多彼此不相关的 Promise,我们可以同时处理它们。

Promise 是并发的,但如你一个一个地等待它们,会太费时间,Promise.all()可以节省很多时间。

记住,Promise.all() 是我们的朋友

  1. const { promisify } = require('util'); 
  2. const sleep = promisify(setTimeout); 
  3.  
  4. async function f1() { 
  5.   await sleep(1000); 
  6.  
  7. async function f2() { 
  8.   await sleep(2000); 
  9.  
  10. async function f3() { 
  11.   await sleep(3000); 
  12.  
  13.  
  14. (async () => { 
  15.   console.time('sequential'); 
  16.   await f1(); 
  17.   await f2(); 
  18.   await f3(); 
  19.   console.timeEnd('sequential');   
  20. })(); 

 上述代码的执行时间约为 6 秒。但如果我们用 Promise.all() 代替它,将减少执行时间。

  1. (async () => { 
  2.     console.time('concurrent'); 
  3.     await Promise.all([f1(), f2(), f3()]); 
  4.     console.timeEnd('concurrent');  
  5.   })(); 

总结

在这篇文章中,我们讨论了使用 Promise 时常犯的五个错误。然而,可能还有很多简单的问题需要仔细解决。

作者:Ravidu Perera 译者:前端小智

来源:medium 原文:https://blog.bitsrc.io/5-common-mistakes-in-using-promises-bfcc4d62657f

 

来源:今日头条内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯