WeakMap+AbortController——优雅地取消请求

AbortController

AbortController可以用来取消fetch请求,MDN给出了简单的demo,就不在此处赘述了。

AbortController 接口表示一个控制器对象,允许你根据需要中止一个或多个 Web 请求。

你可以使用 AbortController() 构造函数创建一个新的 AbortController 对象。使用 AbortSignal 对象可以完成与异步操作的通信。

AbortController取消XHR

AbortController的缺点是无法直接取消XHR,但XHR对象本身就提供了abortapi,我们可以通过监听signal的abort事件,变相实现AbortController取消XHR。

arduino 复制代码
const xhr = new XMLHttpRequest();
xhr.open('GET', '/api/data');
xhr.send();

// 监听 signal
signal.addEventListener('abort', () => {
  xhr.abort(); // 使用 xhr 原生取消
});

axios的请求取消也是基于该思路实现的。

ini 复制代码
/ 创建控制器
const controller = new AbortController();
const signal = controller.signal;

// 发起请求,传入 signal
axios.get('/api/data', { signal })
  .then(response => {
   ...
  })
  .catch(error => {
    ...
  });

// 2 秒后取消请求
setTimeout(() => {
  controller.abort(); // 会触发上方的 catch 分支
}, 2000);

然而项目中请求取消与请求发送往往不在同一个文件中,因此需要缓存请求Promise实例和对应的AbortController实例,这也意味着开发不仅需要关注请求取消,还需要关注这两个实例的清理,以防止内存泄漏,这引入了额外的心智负担。

使用WeakMap缓存请求Promise和AbortController

为了减轻这部分心智负担,我倾向于在项目中使用WeakMap来缓存请求Promise和AbortController。一般来说,项目中不会额外缓存(引用)请求promise,请求完成后,只要该promise"不可达",GC便会回收该实例,对应的abortController也会因为"不可达"而被回收掉。 参考代码如下:

javascript 复制代码
// api.ts
const abortMap = new WeakMap();
window.abortMap = abortMap; //测试gc

const putSth = () => {
  const controller = new AbortController();
  const req = request({ url: "/api/clear", signal: controller.signal }).catch(
    (e) => console.error(e)
  );
  req.t = 1; //测试gc
  abortMap.set(req, controller);
  return req;
};
// component.ts
export function ClearTest() {
  const [visible, setVisible] = useState(true);
  return (
    <div>
      <button
        onClick={() => {
          setVisible((v) => !v);
        }}
      >
        toggle
      </button>
      {visible && <Test></Test>}
    </div>
  );
}
function Test() {
  useEffect(() => {
    const req = putSth();
    return () => {
    //取消请求
      abortMap.get(req)?.abort("cancel");
    };
  }, []);
  return <div>test</div>;
}

内存观察

上面的demo中我加了一些调试代码。对组件进行多次挂载卸载测试,观察控制台可以看到每次请求取消后会打印cancel error观察abortMap,可以明显看到部分请求promise被回收了,程序符合预期:

但也有可能观察到abortMap中存在大量未被回收的promise:

我们可以手动清理垃圾后再观察

手动清理虽然会触发GC,但不一定会清理掉所有垃圾对象(分代垃圾回收)。

通过快照可以看到这些Promise实例的距离都是-,这意味着它们"不可达",如果不在闭包内,将来会被GC清理掉。

经过一段时间后,再次观察可以看到weakMap中所有请求Promise都被回收了:

如果有朋友发现weakMap中的Promise始终无法被回收掉,可能和控制台打印有关系,清理控制台关掉后再打开控制台,支持垃圾回收,来回几次,可能就会看到weakMap被清空了。

相关推荐
辻戋2 小时前
从零实现React Scheduler调度器
前端·react.js·前端框架
徐同保2 小时前
使用yarn@4.6.0装包,项目是react+vite搭建的,项目无法启动,报错:
前端·react.js·前端框架
Qrun3 小时前
Windows11安装nvm管理node多版本
前端·vscode·react.js·ajax·npm·html5
中国lanwp3 小时前
全局 npm config 与多环境配置
前端·npm·node.js
JELEE.4 小时前
Django登录注册完整代码(图片、邮箱验证、加密)
前端·javascript·后端·python·django·bootstrap·jquery
TeleostNaCl6 小时前
解决 Chrome 无法访问网页但无痕模式下可以访问该网页 的问题
前端·网络·chrome·windows·经验分享
前端大卫7 小时前
为什么 React 中的 key 不能用索引?
前端
你的人类朋友7 小时前
【Node】手动归还主线程控制权:解决 Node.js 阻塞的一个思路
前端·后端·node.js
小李小李不讲道理9 小时前
「Ant Design 组件库探索」五:Tabs组件
前端·react.js·ant design
毕设十刻9 小时前
基于Vue的学分预警系统98k51(程序 + 源码 + 数据库 + 调试部署 + 开发环境配置),配套论文文档字数达万字以上,文末可获取,系统界面展示置于文末
前端·数据库·vue.js