0

0

掌握 JavaScript 中的承诺取消

WBOY

WBOY

发布时间:2024-09-12 12:51:00

|

350人浏览过

|

来源于dev.to

转载

作者: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 usage
const 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 保持简单,并专注于表示操作的最终结果:

viable
viable

基于GPT-4的AI非结构化数据分析平台

下载
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.mjs
Received by 1 value: 1
Received by 1 value: 2
Received by 1 value: 3
Observable 1 completed
Received by 2 value: 1
$

结论

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


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

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

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

掌握 JavaScript 中的承诺取消

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

免费试用。

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

相关专题

更多
js获取数组长度的方法
js获取数组长度的方法

在js中,可以利用array对象的length属性来获取数组长度,该属性可设置或返回数组中元素的数目,只需要使用“array.length”语句即可返回表示数组对象的元素个数的数值,也就是长度值。php中文网还提供JavaScript数组的相关下载、相关课程等内容,供大家免费下载使用。

536

2023.06.20

js刷新当前页面
js刷新当前页面

js刷新当前页面的方法:1、reload方法,该方法强迫浏览器刷新当前页面,语法为“location.reload([bForceGet]) ”;2、replace方法,该方法通过指定URL替换当前缓存在历史里(客户端)的项目,因此当使用replace方法之后,不能通过“前进”和“后退”来访问已经被替换的URL,语法为“location.replace(URL) ”。php中文网为大家带来了js刷新当前页面的相关知识、以及相关文章等内容

372

2023.07.04

js四舍五入
js四舍五入

js四舍五入的方法:1、tofixed方法,可把 Number 四舍五入为指定小数位数的数字;2、round() 方法,可把一个数字舍入为最接近的整数。php中文网为大家带来了js四舍五入的相关知识、以及相关文章等内容

706

2023.07.04

js删除节点的方法
js删除节点的方法

js删除节点的方法有:1、removeChild()方法,用于从父节点中移除指定的子节点,它需要两个参数,第一个参数是要删除的子节点,第二个参数是父节点;2、parentNode.removeChild()方法,可以直接通过父节点调用来删除子节点;3、remove()方法,可以直接删除节点,而无需指定父节点;4、innerHTML属性,用于删除节点的内容。

470

2023.09.01

JavaScript转义字符
JavaScript转义字符

JavaScript中的转义字符是反斜杠和引号,可以在字符串中表示特殊字符或改变字符的含义。本专题为大家提供转义字符相关的文章、下载、课程内容,供大家免费下载体验。

388

2023.09.04

js生成随机数的方法
js生成随机数的方法

js生成随机数的方法有:1、使用random函数生成0-1之间的随机数;2、使用random函数和特定范围来生成随机整数;3、使用random函数和round函数生成0-99之间的随机整数;4、使用random函数和其他函数生成更复杂的随机数;5、使用random函数和其他函数生成范围内的随机小数;6、使用random函数和其他函数生成范围内的随机整数或小数。

989

2023.09.04

如何启用JavaScript
如何启用JavaScript

JavaScript启用方法有内联脚本、内部脚本、外部脚本和异步加载。详细介绍:1、内联脚本是将JavaScript代码直接嵌入到HTML标签中;2、内部脚本是将JavaScript代码放置在HTML文件的`<script>`标签中;3、外部脚本是将JavaScript代码放置在一个独立的文件;4、外部脚本是将JavaScript代码放置在一个独立的文件。

652

2023.09.12

Js中Symbol类详解
Js中Symbol类详解

javascript中的Symbol数据类型是一种基本数据类型,用于表示独一无二的值。Symbol的特点:1、独一无二,每个Symbol值都是唯一的,不会与其他任何值相等;2、不可变性,Symbol值一旦创建,就不能修改或者重新赋值;3、隐藏性,Symbol值不会被隐式转换为其他类型;4、无法枚举,Symbol值作为对象的属性名时,默认是不可枚举的。

535

2023.09.20

苹果官网入口直接访问
苹果官网入口直接访问

苹果官网直接访问入口是https://www.apple.com/cn/,该页面具备0.8秒首屏渲染、HTTP/3与Brotli加速、WebP+AVIF双格式图片、免登录浏览全参数等特性。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

10

2025.12.24

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号