前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Node.js v15.x 新特性 — 控制器对象 AbortController

Node.js v15.x 新特性 — 控制器对象 AbortController

作者头像
五月君
发布2021-04-02 01:32:00
1.4K0
发布2021-04-02 01:32:00
举报
文章被收录于专栏:Nodejs技术栈

Node.js v15.0.0 提供了一个全局实用 API AbortController,用于在选定的基于 Promise API 中发出取消信号。无需引入在所有模块中均可使用,该 API 的实现是基于浏览器中的 Web API AbortController。

简单示例

通俗的讲 AbortController 表示一个控制器对象,允许我们根据需要中止一个或多个 Web 请求

下面是一个示例,在 1 秒后会执行 ac.abort() 方法,将会触发 abort 事件,并且仅会触发一次,这可通过 abortSignal.aborted 属性查看前后改变状态。

代码语言:javascript
复制
ac.signal.addEventListener('abort', () => {
  console.log('Aborted!');
  console.log('ac.signal.aborted:', ac.signal.aborted);
}, { once: true });
setTimeout(() => ac.abort(), 1000)
console.log('ac.signal.aborted:', ac.signal.aborted);

中止请求

Node.js 中我们可以选择使用 node-fetch 这个请求处理库,传递 signal 给 fetch。

假设这个请求需要等待 5 秒钟,大约在 2 秒钟后执行 abort() 将会中止这个请求。

代码语言:javascript
复制
const ac = new AbortController();
import fetch from 'node-fetch';

const timer = setTimeout(() => ac.abort(), 2000)
try {
  const { statusText } = await fetch('http://localhost:3000/api', { signal: ac.signal })
  console.log(statusText);
} catch (err) {
  console.log(err.name); // AbortError
} finally {
  clearTimeout(timer);
}

中止 Promise

传递 ac.signal 中止一个正在运行的 Promise,这需要我们为 ac.signal 注册一个 abort 事件,做一些处理。之后在任何地方调用 ac.abort() 中止 Promise。

使用 Promise 表示中止操作的任何 Web 平台 APIs 都必须遵循以下原则:

  • 通过一个 signal 字典成员接受 AbortSignal 对象。
  • 通过 reject 一个带有 "AbortError" DOMException 这个类的 Promise 来表示操作已中止。
  • 检查 AbortSignal 对象的 aborted 标志是否已经被设置,如果是则立即 reject,否则:
  • 使用中止算法机制来观察对 AbortSignal 对象的更改,并以不会导致与其他观察者冲突的方式进行观察。

以下关于 doSomeThingAsync 这个异步 Promise Function 的实现基本上也是遵循的这些规则。

代码语言:javascript
复制
class AbortError extends Error {
  constructor(message) {
    super(message);
    this.name = 'AbortError';
  }
}
function doSomethingAsync({ ac }) {
  return new Promise((resolve, reject) => {
    console.log('task start...');
    if (ac.aborted) {
      return reject(new AbortError('task handler failed', 'AbortError'));
    }

    const timer = setTimeout(() => {
      console.log('task end...');
      resolve(1);
    }, 5000);
    ac.signal.addEventListener('abort', () => {
      clearTimeout(timer);
      reject(new AbortError('task handler failed', 'AbortError'));
    }, { once: true });    
  });
}

setTimeout(() => ac.abort(), 2000)
try {
  await doSomethingAsync({ ac });
} catch (err) {
  console.error(err.name, err.message); // AbortError task handler failed
}

注意:在 Node.js 中目前并没有 DOMException 这个类,我们无法这样做 new DOMException('task handler failed', 'AbortError') 所以我在刚开始先创建了一个 AbortError 类来模拟。

Node.js 中已经有一些异步 API 支持传递 signal,但是它的 DOMException 错误也是在内部通过封装来实现的:

代码语言:javascript
复制
// https://github.com/nodejs/node/blob/f6b1df2226/lib/internal/fs/promises.js#L98

const lazyDOMException = hideStackFrames((message, name) => {
  if (DOMException === undefined)
    DOMException = internalBinding('messaging').DOMException;
  return new DOMException(message, name);
});

// 例如 writeFileHandle
// https://github.com/nodejs/node/blob/f6b1df2226/lib/internal/fs/promises.js#L282
if (signal?.aborted) {
  throw lazyDOMException('The operation was aborted', 'AbortError');
}

Reference

  • https://dom.spec.whatwg.org/#abortcontroller-api-integration
  • https://nodejs.org/docs/latest-v15.x/api/globals.htm
本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2021-03-29,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 Nodejs技术栈 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 简单示例
  • 中止请求
  • 中止 Promise
  • Reference
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档