值得一看
广告
彩虹云商城
广告

热门广告位

优雅地结合 Async/Await 与可取消的 Bluebird Promise

优雅地结合 async/await 与可取消的 bluebird promise

本文旨在解决在使用 Bluebird Promise 实现可取消的异步操作时,async/await 导致的程序卡死问题。我们将探讨如何利用 ES6 内置的 AbortController 来实现 Promise 的取消,并确保程序在取消后能够正常执行后续流程,避免代码阻塞。通过示例代码,我们将展示如何将 AbortController 集成到你的异步函数中,并正确处理取消事件,从而构建更加健壮和可控的异步代码。

在使用 Bluebird 的可取消 Promise 时,结合 async/await 可能会遇到一些问题,尤其是在 Promise 被取消的情况下,程序可能会卡在 await 语句处,导致后续代码无法执行。 这主要是因为取消操作没有正确地通知 await 表达式,使得它一直等待 Promise 的 resolve 或 reject。

解决这个问题的关键在于使用 ES6 内置的 AbortController。AbortController 提供了一种标准的方式来取消异步操作,并且可以与 Promise 结合使用。

使用 AbortController 取消 Promise

以下是一个使用 AbortController 实现可取消 Promise 的示例:

SpeakingPass-打造你的专属雅思口语语料

SpeakingPass-打造你的专属雅思口语语料

使用chatGPT帮你快速备考雅思口语,提升分数

SpeakingPass-打造你的专属雅思口语语料25

查看详情
SpeakingPass-打造你的专属雅思口语语料

let myP = (signal) =>
new Promise((resolve, reject) => {
const tm = setTimeout(function () {
console.log("before resolve");
resolve("Success!"); // Yay! Everything went well!
}, 2500);
// signal.onabort = () => { // 旧写法,不推荐
signal.addEventListener('abort', () => {
console.log('aborted');
clearTimeout(tm);
reject(new Error('Promise Aborted')); // 建议 reject,方便错误处理
// resolve(); // 也可以 resolve,取决于业务需求
});
});
const run = async () => {
console.log("Start");
const controller = new AbortController();
const signal = controller.signal;
const prm = myP(signal);
setTimeout(() => controller.abort(), 1000);
try {
await prm;
console.log("After await prm");
} catch (e) {
console.error("Caught error:", e);
} finally {
console.log("Finished");
}
};
run();

代码解释:

  1. AbortController 的创建: 首先,我们创建一个 AbortController 实例。
  2. signal 的传递: 将 AbortController 的 signal 传递给 myP 函数。signal 对象用于监听取消事件。
  3. addEventListener(‘abort’, …): 在 myP 函数内部,我们使用 signal.addEventListener(‘abort’, …) 监听 abort 事件。 当 AbortController 调用 abort() 方法时,该事件会被触发。
  4. 取消处理: 在 abort 事件处理函数中,我们执行取消操作,包括清除定时器 clearTimeout(tm),并调用 reject(new Error(‘Promise Aborted’)) 来拒绝 Promise。 也可以根据业务需求选择 resolve()。 建议使用 reject,因为这样可以在 catch 块中处理取消错误。
  5. controller.abort() 的调用: 在 run 函数中,我们使用 setTimeout 在 1 秒后调用 controller.abort(),模拟取消操作。
  6. 错误处理: 使用 try…catch 块来捕获可能发生的错误,包括取消导致的错误。

注意事项:

  • reject vs resolve: 当 Promise 被取消时,应该选择 reject 还是 resolve 取决于你的业务逻辑。 如果取消被视为一种错误情况,则应该 reject Promise。 如果取消只是中断了操作,并且不需要进一步处理,则可以 resolve Promise。
  • 资源清理: 在 abort 事件处理函数中,务必清理所有已分配的资源,例如定时器、网络连接等,以避免内存泄漏。
  • signal.onabort vs signal.addEventListener(‘abort’, …): 建议使用 addEventListener,因为它允许你注册多个取消处理函数。 避免使用 signal.onabort = …,因为它只能注册一个处理函数。
  • 多个 Promise 的取消: 可以使用同一个 AbortController 的 signal 来取消多个 Promise。 这在需要同时取消多个相关操作时非常有用。

总结

AbortController 提供了一种简洁而强大的方式来取消异步操作,并且可以与 async/await 和 Promise 无缝集成。 通过使用 AbortController,你可以构建更加健壮和可控的异步代码,避免程序卡死,并有效地管理资源。 记得在取消处理函数中清理资源并根据业务需求选择 reject 或 resolve Promise。

相关标签:

es6 ai es6 try catch Error signal 对象 事件 promise 异步
温馨提示: 本文最后更新于2025-10-08 10:40:16,某些文章具有时效性,若有错误或已失效,请在下方留言或联系在线客服
文章版权声明 1 本网站名称: 创客网
2 本站永久网址:https://new.ie310.com
1 本文采用非商业性使用-相同方式共享 4.0 国际许可协议[CC BY-NC-SA]进行授权
2 本站所有内容仅供参考,分享出来是为了可以给大家提供新的思路。
3 互联网转载资源会有一些其他联系方式,请大家不要盲目相信,被骗本站概不负责!
4 本网站只做项目揭秘,无法一对一教学指导,每篇文章内都含项目全套的教程讲解,请仔细阅读。
5 本站分享的所有平台仅供展示,本站不对平台真实性负责,站长建议大家自己根据项目关键词自己选择平台。
6 因为文章发布时间和您阅读文章时间存在时间差,所以有些项目红利期可能已经过了,能不能赚钱需要自己判断。
7 本网站仅做资源分享,不做任何收益保障,创业公司上收费几百上千的项目我免费分享出来的,希望大家可以认真学习。
8 本站所有资料均来自互联网公开分享,并不代表本站立场,如不慎侵犯到您的版权利益,请联系79283999@qq.com删除。

本站资料仅供学习交流使用请勿商业运营,严禁从事违法,侵权等任何非法活动,否则后果自负!
THE END
喜欢就支持一下吧
点赞13赞赏 分享
评论 抢沙发

请登录后发表评论

    暂无评论内容