掌握 JavaScript 中的承诺取消

作者:rosario de chiara✏️

在 javascript 中,promises 是处理异步操作的强大工具,在 ui 相关事件中特别有用。它们代表的值可能无法立即获得,但会在未来某个时刻得到解决。

promise 允许(或应该允许)开发人员在处理 api 调用、用户交互或动画等任务时编写更清晰、更易于管理的代码。通过使用 .then()、.catch() 和 .finally() 等方法,promises 能够以更直观的方式处理成功和错误场景,避免臭名昭著的“回调地狱”。

在本文中,我们将使用新的(2024 年 3 月)promise.withresolvers() 方法,该方法允许您通过返回一个包含三件事的对象来编写更干净、更简单的代码:一个新的 promise 和两个函数,一个用于解析 promise另一个拒绝它,因为这是最近的更新,您将需要最新的 node 运行时 (v>22) 来执行本文中的示例。

比较新旧 javascript promise 方法

在以下两个功能等效的代码块中,我们可以比较旧方法和分配方法来解析或拒绝 promise 的新方法:

let resolve, reject;const promise = new promise((res, rej) => {  resolve = res;  reject = rej;});math.random() > 0.5 ? resolve("ok") : reject("not ok");

在上面的代码中,你可以看到 promise 最传统的用法:实例化一个新的 promise 对象,然后在构造函数中,你必须分配resolve和reject这两个函数,这两个函数将在以下情况下被调用:需要。

立即学习“Java免费学习笔记(深入)”;

在下面的代码片段中,相同的代码块已使用新的 promise.withresolvers() 方法重写,并且看起来更简单:

const { promise, resolve, reject } = promise.withresolvers();math.random() > 0.5 ? resolve("ok") : reject("not ok");

在这里您可以看到新方法的工作原理。它返回 promise,您可以在该 promise 上调用 .then() 方法和两个函数:resolve 和reject。

传统的 promise 方法将创建和事件处理逻辑封装在单个函数中,如果多个条件或代码的不同部分需要解析或拒绝 promise,这可能会受到限制。

相比之下,promise.withresolvers() 通过将 promise 的创建与解析逻辑分离,提供了更大的灵活性,使其适合管理复杂的条件或多个事件。然而,对于简单的用例,传统方法对于那些习惯标准承诺模式的人来说可能更简单、更熟悉。

真实示例:调用 api

我们现在可以在更现实的示例上测试新方法。在下面的代码中,您可以看到一个 api 调用的简单示例:

function fetchdata(url) {    return new promise((resolve, reject) => {        fetch(url)            .then(response => {                // check if the response is okay (status 200-299)                if (response.ok) {                    return response.json(); // parse json if response is okay                } else {                    // reject the promise if the response is not okay                    reject(new error('api invocation failed'));                }            })            .then(data => {                // resolve the promise with the data                resolve(data);            })            .catch(error => {                // catch and reject the promise if there is a network error                reject(error);            });    });}// example usageconst apiurl = '';fetchdata(apiurl)    .then(data => {        // handle the resolved data        console.log('data received:', data);    })    .catch(error => {        // handle any errors that occurred        console.error('error occurred:', error);    });

fetchdata 函数旨在获取 url 并返回 promise,该 promise 使用 fetch api 处理 api 调用。它通过检查响应状态是否在 200-299 范围内(表示成功)来处理响应。

如果成功,响应将被解析为 json,并且 promise 将使用结果数据进行解析。如果响应不成功,则 promise 会被拒绝,并显示相应的错误消息。此外,该函数还包括错误处理以捕获任何网络错误,如果发生此类错误则拒绝 promise。

该示例演示了如何使用此函数,展示了如何使用 .then() 块管理已解析的数据并使用 .catch() 块处理错误,确保成功的数据检索和错误得到适当的管理。

在下面的代码中,我们使用新的 promise.withresolvers() 方法重写了 fetchdata() 函数:

function fetchdata(url) {    const { promise, resolve, reject } = promise.withresolvers();    fetch(url)        .then(response => {            // check if the response is okay (status 200-299)            if (response.ok) {                return response.json(); // parse json if response is okay            } else {                // reject the promise if the response is not okay                reject(new error('api invocation failed'));            }        })        .then(data => {            // resolve the promise with the data            resolve(data);        })        .catch(error => {            // catch and reject the promise if there is a network error            reject(error);        });    return promise;}

如您所见,上面的代码更具可读性,并且 promise 对象的作用很明确:fetchdata 函数将返回一个将成功解析或将失败的 promise,在每种情况下调用正确的方法。您可以在名为 api.inspiration.{old|new}.js.

的存储库中找到上面的代码

承诺取消

以下代码探讨了如何实现 promise 取消方法。如您所知,您无法取消 javascript 中的 promise。 promise 代表异步操作的结果,它们被设计为一旦创建就解决或拒绝,没有内置机制来取消它们。

出现这个限制是因为 promise 有定义的状态转换过程;它们一开始处于待定状态,一旦解决,就无法更改状态。它们旨在封装操作的结果,而不是控制操作本身,这意味着它们不能影响或取消底层过程。这种设计选择使 promise 保持简单,并专注于表示操作的最终结果:

const cancellablepromise = () => {    const { promise, resolve, reject } = promise.withresolvers();    promise.cancel = () => {        reject("the promise got cancelled");    };    return promise;};

在上面的代码中,您可以看到名为cancellablepromise的对象,它是一个带有附加cancel()方法的promise,正如您所看到的,它只是强制调用reject方法。这只是语法糖,不会取消 javascript promise,尽管它可能有助于编写更清晰的代码。

另一种方法是使用 abortcontroller 和 abortsignal,它们可以绑定到底层操作(例如 http 请求)以在需要时取消它。从文档中,您可以看到 abortcontroller 和 abortsignal 方法是我们在上面的代码中实现的更具表现力的实现:一旦调用 abortsignal,promise 就会被拒绝。

另一种方法是使用 rxjs 等反应式编程库,它提供了 observable 模式的实现,对异步数据流进行更复杂的控制,包括取消功能。

observables 和 promise 之间的比较

在谈到实际用例时,promise 非常适合处理单个异步操作,例如从 api 获取数据。相比之下,observables 非常适合管理数据流,例如用户输入、websocket 事件或 http 响应,其中可能会随着时间的推移发出多个值。

我们已经澄清,一旦启动,promise 就无法取消,而 observables 允许通过取消订阅流来取消。总的想法是,使用 observables,您可以拥有与对象可能交互的显式结构:

你创建一个 observable,然后所有的 observable 都可以订阅它 observable 执行其工作,改变状态并发出事件。所有观察者都会收到更新——这是与承诺的主要区别。 promise 只能被解析一次,而 observables 只要有 observers 就可以不断发出事件 一旦观察者对 observable 中的事件不感兴趣,它就可以取消订阅,释放资源

这在下面的代码中得到了演示:

import { observable } from 'rxjs';const observable = new observable(subscriber => {  subscriber.next(1);  subscriber.next(2);  subscriber.next(3);  subscriber.complete();});const observer = observable.subscribe({  next(x) { console.log('received value:', x); },  complete() { console.log('observable completed'); }});observer.unsubscribe();

这段代码不能用 promise 重写,因为 observable 返回三个值,而 promise 只能解析一次。

为了进一步试验取消订阅方法,我们可以添加另一个将使用 takewhile() 方法的观察者:它将让观察者等待值匹配特定条件;例如,在下面的代码中,当值不为 2 时,它会不断接收来自 observable 的事件:

import { observable, takewhile } from 'rxjs';const observable = new observable(subscriber => {  subscriber.next(1);  subscriber.next(2);  subscriber.next(3);  subscriber.complete();});const observer1 = observable.subscribe({  next(x) { console.log('received by 1 value:', x); },  complete() { console.log('observable 1 completed'); }});const observer2 = observable.pipe(  takewhile(value => value != "2")).subscribe(value => console.log('received by 2 value:', value));

在上面的代码中,observer1 与我们已经看到的相同:它只会订阅并持续接收来自 observable 的所有事件。第二个,observer2,将在条件匹配时从 observable 接收元素。在本例中,这意味着该值不同于 2。

从执行中,你可以看到两种不同的机制是如何工作的:

$ node observable.mjsReceived by 1 value: 1Received by 1 value: 2Received by 1 value: 3Observable 1 completedReceived by 2 value: 1$

结论

在本文中,我们研究了在 javascript 中分配 promise 的新机制,并列出了在 promise 完成之前取消 promise 的一些可能方法。我们还将 promises 与 observable 对象进行了比较,后者不仅提供 promises 的功能,还通过允许多次发出事件和适当的取消订阅机制来扩展它们。

logrocket:通过了解上下文更轻松地调试 javascript 错误

调试代码始终是一项乏味的任务。但你越了解自己的错误,就越容易纠正它们。

logrocket 允许您以新的、独特的方式理解这些错误。我们的前端监控解决方案跟踪用户与 javascript 前端的互动,使您能够准确查看用户的操作导致了错误。

掌握 JavaScript 中的承诺取消

logrocket 记录控制台日志、页面加载时间、堆栈跟踪、带有标头 + 正文的慢速网络请求/响应、浏览器元数据和自定义日志。了解 javascript 代码的影响从未如此简单!

免费试用。

以上就是掌握 JavaScript 中的承诺取消的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1491345.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月19日 13:53:00
下一篇 2025年12月19日 13:53:11

相关推荐

  • RemoveCookieWall,Firefox 扩展

    您是否厌倦了网站上流行的横幅,以便您接受第三方 cookie 或结账?在这篇文章中,我解释了如何制作(并发布)一个 firefox 扩展以避免大多数网站出现这种情况 信息 此扩展的代码发布于 https://github.com/jagedn/removecookiewall-addon您可以从 h…

    好文分享 2025年12月19日
    000
  • JavaScript 中的闭包

    你好, 在这篇文章中,让我们学习闭包。 闭包在 javascript 中至关重要,因为它们允许函数从其父作用域访问变量,即使在父函数关闭之后也是如此。这对于需要随着时间的推移记住数据的函数至关重要,例如回调函数或维护状态。这里要记住的一点是父作用域中未使用的变量将被垃圾收集。 定义:闭包是捆绑在一起…

    2025年12月19日
    000
  • 带 Fetch 的 AbortController

    javascript 中的 abortcontroller 是一个实用程序,用于取消或中止异步操作,例如获取请求或事件侦听器等其他任务,这些任务可能需要一些时间才能完成。它允许您停止不再需要的操作,这对于提高性能和管理资源很有用。 示例用例: // Create an AbortController…

    2025年12月19日
    000
  • Effect-TS 中的过滤选项:实用指南

    effect-ts 提供了各种方法来过滤选项内的值,允许您对可选值应用转换、谓词或检查。这些函数有助于确保仅保留相关数据,同时丢弃 none 值或不满足指定条件的值。在本文中,我们将探讨用于过滤选项的四个关键函数:o.partitionmap、o.filtermap、o.filter 和 o.exi…

    2025年12月19日
    000
  • 什么是 Web Worker 以及如何在 NextJS 中使用它

    先决条件 reactjs/nextjs 基础知识 什么是网络工作者 javascript 是一种单线程语言,它使用的线程称为主线程浏览器实际上使用其他线程来自浏览器 api 的 web worker 是您使用 javascript 创建和注册附加线程的一种方式 当您只能在主线程上工作时,为什么还要创…

    2025年12月19日
    000
  • 在 Effect-TS 中组合选项:实用指南

    effect-ts 提供了几种在函数式编程上下文中组合可选值或选项的强大方法。无论您想要将多个选项配对在一起还是将选项内的函数应用于其他值,该库都提供了多种方法来简化这些操作。在本文中,我们将探讨组合选项的四个关键函数:o.product、o.productmany、o.all 和 o.ap。 示例…

    2025年12月19日
    000
  • 了解 React Cache 功能

    随着react生态系统的不断扩大,优化数据获取的更强大的工具之一就是缓存功能。此内置功能允许您执行许多操作,例如有效管理和存储服务器数据、减少冗余网络请求以及提高整体应用程序性能。 在本文中,我们将了解 react 中的缓存功能、它的好处以及如何使用它。 什么是react缓存功能 react 发布的…

    2025年12月19日
    000
  • 释放 WordPress 的力量:构建令人惊叹的网站的指南

    在快速发展的 Web 开发世界中,WordPress 仍然是一次又一次重复使用的珍贵核心,用于将网站的各种性质构建为强大而灵活的解决方案,从而构建易于管理的出色网站。无论是强大的开发人员、企业主还是数字空间 WordPress 的新手。这个博客将帮助您完成整个过程,从了解功能和性能到设置您的网站和 …

    2025年12月19日
    000
  • Nodejs 版本管理器终极指南:NVM、NVS、fnm、Volta 和 asdf |第 1 部分

    介绍 嘿!如果您曾经使用 node.js 进行过开发,您就会明白,对于项目来说,经常需要使用不同的版本。也许一个项目可以在版本 10 上运行,另一个项目可以在版本 14 上运行,而一个新项目需要最新版本,比如 20。随着每个新版本的新功能的出现,一系列新的挑战也随之出现。这些是与库和框架的兼容性、新…

    2025年12月19日
    000
  • 揭秘合并排序:分治排序初学者指南

    归并排序由约翰·冯·诺依曼于 1945 年提出,主要是为了提高大型数据集的排序效率。冯·诺依曼的算法旨在使用分而治之的方法提供一致且可预测的排序过程。这种策略允许归并排序有效地处理小型和大型数据集,保证在所有情况下都能实现稳定的排序,时间复杂度为 o(n log n)。 合并排序采用分而治之方法,将…

    2025年12月19日
    000
  • 异步

    javascript 中的异步编程 允许您执行 api 调用、读取文件或查询数据库等任务,而不会阻塞其他代码的执行。这在 javascript 中至关重要,尤其是在 web 开发中,响应能力和性能是关键。 关键概念 1。回调: 作为参数传递给另一个函数的函数,该函数在异步操作完成后执行。 示例: f…

    2025年12月19日
    000
  • 安全分配

    今天关于 javascript 中安全赋值运算符 (?=) 的新提案引起了热烈讨论。我喜欢 javascript 随着时间的推移而不断改进,但这也是我最近在一些情况下遇到的问题。我应该将一个快速示例实现作为函数,对吧? 如果您还没有阅读该提案,以下是其建议: const [error, value]…

    2025年12月19日
    000
  • useSyncExternalStoreExports 状态源码解释

    在本文中,我们将了解 zustand 如何在其[源代码]中使用 usesyncexternalstoreexports。 usesyncexternalstoreexports 是从 use-sync-external-store/shim/with-selector 导入的。 use-sync-e…

    2025年12月19日
    000
  • 使用可重用列表组件扩展 React 应用程序

    在 react 中构建可扩展的应用程序需要的不仅仅是可靠的业务逻辑。随着应用程序的发展,组件的架构对于其可维护性、性能和灵活性起着重要作用。许多 web 应用程序中的基本任务之一是处理数据列表。无论是渲染产品列表、表格还是仪表板,您经常会遇到需要可重复和可重用的列表结构的场景。 通过构建可重用的列表…

    2025年12月19日
    000
  • 简化 React 状态管理:处理状态的最佳实践

    有时候,我们必须在react状态下管理状态。比如我们有一个提交表单,我们要管理表单的状态。 表达状态的方式有很多种。我会介绍一下表达身份的坏例子。 坏榜样 1、用物体来表达状态。 const page = () => { const [status, setstatus] = usestate…

    2025年12月19日
    000
  • Redis:内存数据结构存储终极指南

    redis 是不断发展的数据管理和存储领域中广泛使用的技术。 redis 被公认为内存中数据结构存储,它提供了广泛的功能,使其成为从缓存到实时分析等各种应用程序的标准基础。这个综合教程将介绍 redis 是什么、它的核心功能、用例以及如何开始。 什么是redis? redis代表远程字典服务器;它是…

    2025年12月19日
    000
  • esser-已知的 HTML 属性可增强用户体验

    html 是一种用于创建网页的强大语言,虽然大多数开发人员都熟悉常用的 html 属性,但还有一些鲜为人知的属性可以提供附加功能并增强用户体验。 1. 虚拟键盘的enterkeyhint属性 在为移动设备设计 web 表单时,考虑用户与之交互的虚拟键盘非常重要。 enterkeyhint 属性是一个…

    2025年12月19日
    000
  • 如何使用 Tailwind CSS 和 JavaScript 创建基本绘图工具并保存为 PNG

    今天,我们将使用 Tailwind CSS 和 JavaScript 创建一个简单的绘图工具。我们将使用 canvas 元素进行绘图,并使用 toDataURL 方法将图像保存为 PNG。 绘图工具是一款数字应用程序,可让用户创建和编辑视觉内容。它通常提供一个画布,用于使用画笔或钢笔等工具徒手绘画,…

    2025年12月19日
    000
  • 完整的 redux 工具包(第 4 部分)

    第 4 部分:rtk 查询中的高级主题。 本部分将重点介绍 rtk 查询中的高级功能和用例,包括自定义查询、处理身份验证、乐观更新和性能优化。 第 4 部分:rtk 查询中的高级主题 1. 高级rtk查询概念介绍 在上一部分中,我们介绍了使用 rtk 查询来获取和变异数据的基础知识。现在,我们将深入…

    2025年12月19日 好文分享
    000
  • NODE面试问题

    1. 什么是 Node.js? 答案: Node.js 是一个基于 Chrome 的 V8 JavaScript 引擎构建的运行时环境,允许开发者在服务器端运行 JavaScript 代码。它专为构建可扩展的网络应用程序而设计,并支持非阻塞、事件驱动的架构。 2. Node.js 和 JavaScr…

    2025年12月19日
    000

发表回复

登录后才能评论
关注微信