項(xiàng)目中如何使用axios過濾多次重復(fù)請求詳解
一、前言:
我們在web應(yīng)用開發(fā)過程當(dāng)中,經(jīng)常會(huì)遇到一個(gè)時(shí)刻發(fā)起了多個(gè)請求的場景
這個(gè)情況下,我們通常的做法有兩種:
- 可以在請求時(shí)show一個(gè)loading,阻止用戶操作。
- 或者人為加個(gè)變量,做一個(gè)請求的節(jié)流
我們的項(xiàng)目中,目前大部分情況也是采用以上兩種方式做的。今天來介紹一個(gè)新的方式。
二、CancelToken類
我們之前實(shí)例化一個(gè)Promise,這個(gè)對象是否成功與否,是無法在函數(shù)外部決定的,這里邊使用要用到一個(gè)小竅門,可以讓一個(gè)promise 和resolve分離。任何時(shí)機(jī)都可以觸發(fā)resolve:
// 一個(gè)promise let resolvePromise let p = new Promise((resolve, reject) => { resolvePromise = resolve }) // 這樣在外部執(zhí)行 resolvePromise()
ok,有了這個(gè)前提,我們需要借助axios.CancelToken這個(gè)類。
這個(gè)類相當(dāng)于在每次請求的時(shí)候開啟另一個(gè)promise和當(dāng)前的請求形成一個(gè)promise.race(請求p1,取消請求p2),在promise中的resolve方法賦值給了一個(gè)外部的變量去接收。我們可以根據(jù)需求,人為決定是否取消前次請求。其實(shí)這就類似,原來我們寫fetch封裝接口超時(shí)的Promise.race類似。
cancelToken中也提供了相應(yīng)的static方法source用來生成一個(gè)cancelToken和一個(gè)cancel方法其實(shí)就是這個(gè)promise的一個(gè)resolve。
CancelToken.source = function source() { var cancel; // var token = new CancelToken(function executor(c) { cancel = c; }); return { token: token, cancel: cancel, };
根據(jù)我們常用的的緩存方式,我們可以聲明一個(gè)map來存儲(chǔ)每次請求的url,同時(shí)存儲(chǔ)對應(yīng)的cancel方法。
// 聲明一個(gè)全局map const pendingHttp = new Map() // axios中內(nèi)置的CancelToken類 const CancelToken = axios.CancelToken function addApi (config) { config.cancelToken = new CancelToken((cancel) => { const url = config.url console.log(pendingHttp) if (!pendingHttp.has(url)) { pendingHttp.set(url, cancel) } }) } function cancelApi (config) { const url = config.url if (pendingHttp.has(url)) { // 如果在 pending 中存在當(dāng)前請求標(biāo)識,需要取消當(dāng)前請求,并且移除 const cancel = pendingHttp.get(url) cancel(url + '取消了') pendingHttp.delete(url) // 清空當(dāng)前url的緩存 } }
- 要特殊注意,要想取消掉一個(gè)請求,需要在config上添加cancelToken這個(gè)屬性賦值為CancelToken的實(shí)例。否則cancel不掉。
就像操作定時(shí)器一樣,要先嘗試取消上一次,然后再開啟下一次
httpService.interceptors.request.use(config => { cancelApi(config) addApi(config) // 本地調(diào)試的時(shí)候,是跨域的情況,加請求頭會(huì)有限制(此處是項(xiàng)目代碼無關(guān)緊要) const { headers = {} } = config; const { globalObj = {} } = window Object.assign(headers, globalObj, { from }) return config }, error => { console.log(error) return Promise.reject(error) })
然后還有一種可能性,第一次請求已經(jīng)返回了,又發(fā)起了相同的一次請求,所以在response里邊也要cancelApi一下。
httpService.interceptors.response.use( response => { cancelApi(response.config) sentryCatchApi(response) }, error => { // 請求超時(shí) if (error.message.includes('timeout')) { // 判斷請求異常信息中是否含有超時(shí)timeout字符串 Toast.error({ text: '網(wǎng)頁請求超時(shí),請刷新重試~' }) } sentryCatchApi(error) return Promise.reject(error) } )
我們需要注意一點(diǎn),cancel其實(shí)就是resolve,我們cancel執(zhí)行時(shí)候傳入的參數(shù),會(huì)最終在response的error回調(diào)中,作為參數(shù)返回,這樣我們的捕捉錯(cuò)誤的方法可能會(huì)有報(bào)錯(cuò)。
// 假設(shè)我們的error方法是這樣封裝的。 來看一下sentryCatchApi error => { sentryCatchApi(error) return Promise.reject(error) } // 由于這個(gè)方法需要接收一個(gè)對象,但是我們cancel取消請求的情況下,返回的是一個(gè)字符串,這時(shí)就報(bào)錯(cuò)了。 function sentryCatchApi (res) { try { res = res || {} const resData = res.data || {} Sentry.captureException(JSON.stringify(resData)) console.log(` 獲取數(shù)據(jù)失敗: 請?jiān)跒g覽器中打開進(jìn)入 webview的地址,并粘貼出來,便于問題排查 :接口相關(guān)信息: 接口地址:${res.config.url}, 接口返回值:code:${resData.code}, message:${resData.message}, data:${JSON.stringify(resData.data)} `) } catch (err) { console.error(err) } }
需要使用isCancel這個(gè)api
error => { if (axios.isCancel(error)) return console.log('請求被取消了', error.message) sentryCatchApi(error) return Promise.reject(error) }
最終效果
控制臺(tái)也沒有任何報(bào)錯(cuò)啦。(后續(xù)完善到項(xiàng)目中)
總結(jié)
到此這篇關(guān)于項(xiàng)目中如何使用axios過濾多次重復(fù)請求的文章就介紹到這了,更多相關(guān)axios過濾多次重復(fù)請求內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
vue工程全局設(shè)置ajax的等待動(dòng)效的方法
這篇文章主要介紹了vue工程全局設(shè)置ajax的等待動(dòng)效的方法,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2019-02-02淺談Vue3.0新版API之composition-api入坑指南
這篇文章主要介紹了Vue3.0新版API之composition-api入坑指南,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-04-04淺談vue項(xiàng)目4rs vue-router上線后history模式遇到的坑
今天小編就為大家分享一篇淺談vue項(xiàng)目4rs vue-router上線后history模式遇到的坑,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-09-09vue-video-player 解決微信自動(dòng)全屏播放問題(橫豎屏導(dǎo)致樣式錯(cuò)亂問題)
這篇文章主要介紹了vue-video-player 解決微信自動(dòng)全屏播放問題,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-02-02vue-router中query和params的區(qū)別解析
vue-router是Vue.js官方的路由插件,它和vue.js是深度集成的,適合用于構(gòu)建單頁面應(yīng)用,這篇文章主要介紹了vue-router中query和params的區(qū)別 ,需要的朋友可以參考下2022-10-10vue開發(fā)中后臺(tái)系統(tǒng)復(fù)雜表單優(yōu)化技巧
這篇文章主要為大家介紹了vue開發(fā)中后臺(tái)系統(tǒng)復(fù)雜表單的優(yōu)化技巧,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-07-07Vue-cli3項(xiàng)目配置Vue.config.js實(shí)戰(zhàn)記錄
這篇文章主要給大家介紹了關(guān)于Vue-cli3項(xiàng)目配置Vue.config.js的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2018-07-07詳解Vue自定義指令如何實(shí)現(xiàn)處理圖片加載失敗的碎圖
這篇文章主要介紹了詳解Vue自定義指令如何實(shí)現(xiàn)處理圖片加載失敗的碎圖,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)吧2023-02-02