取消正在运行的Promise技巧详解
前言
最近项目当中小伙伴遇到一个很奇怪的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 ( <div> <button onClick={go}>go to</button> Test </div> ); } export default Test;
我们进入Test组件后,马上点击go to按钮,几秒之后页面还会跳转到test1页面。
经分析,我们应该在离开的时候要取消请求和取消Promise让后续的业务代码不在执行,取消请求比较简单,一般的库都支持,我们来说下怎么取消Promise.
CancelablePromise (取消Promise)
我们知道Promise是没有提供取消或者终止的操作。但我们在开发过程中会遇到。我们可以参考和借助AbortController来实现。
class CancelablePromise<T> { /** * 构造器 * @param executor Promise中的 executor * @param abortSignal AbortController中的signal对象 * @returns */ constructor(executor: (resolve: (value: T | PromiseLike<T>) => void, reject: (reason?: any) => void) => void, abortSignal: AbortSignal) { // 记录reject和resolve方法 let _reject: any = null; let _resolve: any = null; let _isExecResolve = false; // 创建和执行Promise const cancelablePromise = new Promise<T>((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 ( <div> <button onClick={go}>go to</button> Test </div> ); } 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运行取消的资料请关注猪先飞其它相关文章!
原文出处:https://juejin.cn/post/7012487154714542111
相关文章
- Promise是异步编程的一种解决方案,在ES6中Promise被列为了正式规范,统一了用法,原生提供了Promise对象。接下来通过本文给大家介绍Promise的介绍及基本用法,感兴趣的朋友一起看看吧...2021-10-21
- 这篇文章主要介绍了封装 axios+promise通用请求函数操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-08-12
jQuery的promise与deferred对象在异步回调中的作用
这篇文章主要介绍了jQuery的promise与deferred对象在异步回调中的作用,需要的朋友可以参考下...2016-05-05- 这篇文章主要给大家介绍了关于ES6学习教程之Promise用法的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2020-11-23
详解JavaScript异步编程中jQuery的promise对象的作用
这篇文章主要介绍了JavaScript异步编程中jQuery的promise对象的作用,同时也谈到了js的Dojo框架中promise模式的实现,需要的朋友可以参考下...2016-05-05- 这篇文章主要介绍了在vue项目中promise解决回调地狱和并发请求的问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-11-09
javascript下使用Promise封装FileReader
这篇文章主要介绍了javascript下使用Promise封装FileReader,需要的朋友可以参考下...2016-02-21- 这篇文章主要介绍了jQuery中的Deferred和promise 的区别的相关资料,需要的朋友可以参考下...2016-04-06
深入解析jQuery中Deferred的deferred.promise()方法
这篇文章主要介绍了jQuery中Deferred的deferred.promise()方法,提醒千万要注意deferred.promise()与jQuery的.promise() 实例方法的区别,需要的朋友可以参考下...2016-05-05- 这篇文章主要为大家介绍了JavaScript中Promise的使用,本文先介绍 Promises 相关规范,然后再通过解读一个迷你的 Promises 以加深理解,感兴趣的小伙伴们可以参考一下...2016-01-21
- 这篇文章主要介绍了JS异步代码单元测试之神奇的Promise,对异步感兴趣的同学,可以参考下...2021-05-07
- 这篇文章主要为大家介绍了取消正在运行的Promise技巧详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪...2022-06-08
- Promise 提供了一种优雅的方法来处理 JS 中的异步操作。这也是避免“回调地狱”的解决方案。然而,并没有多少开发人员了解其中的内容。因此,许多人在实践中往往会犯错误。在本文中,介绍一下使用 promise 时的五个常见错误,希望大家能够避免...2022-11-11
- 本文详细讲解了JS异步编程之Promise对象,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下...2022-06-26
- 这篇文章主要为大家介绍了Promise对象all与race方法手写示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪...2022-12-12
JavaScript详解使用Promise处理回调地狱与async await修饰符
这篇文章主要介绍了JavaScript使用Promise处理回调地狱与async await修饰符,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2022-07-19- 这篇文章主要介绍了javascript中Promise使用详解,关于Promise的使用相关的知识点下面文章内容介绍详细具有一定的参考价值,需要的小伙伴可以参考一下...2022-04-12
Promise.race和Promise.any使用示例详解
这篇文章主要为大家介绍了如何使用Promise.race()和Promise.any()示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪...2022-11-11- 本文详细讲解了ES6的Promise用法,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下...2021-12-15