取消正在运行的Promise技巧详解
目录
- 前言
- 代码案例
- CancelablePromise (取消Promise)
- 使用
- 多个Promise链式调用
- 在async和await中使用
- 结束语
前言 最近项目当中小伙伴遇到一个很奇怪的bug,进入一个页面后,快速切换到其它页面,会跳转到403页面。经过一段时间和小伙伴的排查,发现那个页面有个接口请求响应时间比较长,请求后还有一些业务处理。
等我们切换到其它页面,这个请求完成后还会处理剩下的业务,导致出错。
代码案例 项目当中有很多业务,我们用一些简单代码复现下这个问题。
import React, { useEffect } from 'react'; import { history } from 'umi'; const Test = props => {useEffect(() => {new Promise((resolve, reject) => {// 模拟接口请求时间setTimeout(() => {resolve()}, 4000); }).then(res => {return new Promise((resolve1) => {// 模拟接口请求时间setTimeout(() => {resolve1()}, 1000); })}).then(() => {// Promise 执行完后页面跳转history.push('/test1')})}, []); const go = () => {history.push('/user/login')}return (Test); }export default Test;
我们进入Test组件后,马上点击go to按钮,几秒之后页面还会跳转到test1页面。
经分析,我们应该在离开的时候要取消请求和取消Promise让后续的业务代码不在执行,取消请求比较简单,一般的库都支持,我们来说下怎么取消Promise.
CancelablePromise (取消Promise) 我们知道Promise是没有提供取消或者终止的操作。但我们在开发过程中会遇到。我们可以参考和借助AbortController来实现。
class CancelablePromise{/*** 构造器* @param executor Promise中的 executor* @param abortSignal AbortController中的signal对象* @returns */constructor(executor: (resolve: (value: T | PromiseLike ) => void, reject: (reason?: any) => void) => void, abortSignal: AbortSignal) {// 记录reject和resolve方法let _reject: any = null; let _resolve: any = null; let _isExecResolve = false; // 创建和执行Promiseconst cancelablePromise = new Promise ((resolve, reject) => {_reject = reject; _resolve = (value: T) => {_isExecResolve = true; resolve(value); }; return executor(_resolve, reject); }); // 监听Signal的abourt事件abortSignal.addEventListener('abort', () => {if (_isExecResolve) {return; }// 抛出错误const error = new DOMException('user cancel promise', CancelablePromise.CancelExceptionName ); _reject( error ); } ); return cancelablePromise; }// 取消后抛出的异常名称static CancelExceptionName = 'CancelablePromise AbortError'; }export default CancelablePromise;
使用 下面我们该造下之前的代码,用我们封装的CancelablePromise,可以让Promise可取消。
多个Promise链式调用
import React, { useEffect } from 'react'; import { history } from 'umi'; import CancelablePromise from '../utils/CancelablePromise'; const Test = props => {useEffect(() => {let abortController = new AbortController(); new CancelablePromise((resolve, reject) => {// 模拟接口请求时间setTimeout(() => {resolve()}, 4000); }, abortController.signal).then(res => {return new CancelablePromise((resolve1) => {// 模拟接口请求时间setTimeout(() => {resolve1()}, 1000); }, abortController.signal)}).then(() => {// Promise 执行完后页面跳转history.push('/test1')})return () => {// 取消请求abortController.abort(); }}, []); const go = () => {history.push('/user/login')}return (Test); }export default Test;
在async和await中使用
import React, { useEffect } from 'react'; import { history } from 'umi'; import CancelablePromise from '../utils/CancelablePromise'; const Test = props => {useEffect(() => {let abortController = new AbortController(); const exec = async function () {try {await new CancelablePromise((resolve) => {setTimeout(() => {resolve(); }, 5000); }, abortController.signal, 'one')await new CancelablePromise((resolve1) => {setTimeout(() => {resolve1(); }, 5000); }, abortController.signal, 'del')history.push('/test')} catch (error) {// 取消之后会抛出异常if (CancelablePromise.CancelExceptionName === error.name) {console.log('promise 终止了。。。')}}}exec(); return () => {// 取消请求abortController.abort(); }}, []); }
Promise取消之后会抛出异常,如果需要在抛出异常之后做处理,可以通关对应的异常名称做判断。
if (CancelablePromise.CancelExceptionName === error.name) {console.log('promise 终止了。。。')}
结束语 【取消正在运行的Promise技巧详解】Promise的取消在业务当中用到的地方比较多,更多关于Promise运行取消的资料请关注脚本之家其它相关文章!
推荐阅读
- 计算机中的算术运算
- 投稿|“影舞者”还是“过河卒”?鼎桥的命运握谁手中
- 运用注册表处理win7电脑无法识别U盘设置图文详细教程
- 怎样运用u盘保存上网登录账号密码完成迅速登录?
- 电脑局限运用U盘怎样办?
- 如何加速U盘运行?
- 运用U盘复制文件时电脑自动重启处理办法
- 利用Flow每天获取Flow的状态及运行记录
- 产业互联网|货运IoTSaaS两独角兽G7&E6宣布合并,将保留两个利润中心
- oauth2.0授权码模式详解