axios取消請(qǐng)求CancelToken的用法示例代碼
axios中取消請(qǐng)求情況分為1:取消該請(qǐng)求之后的相同請(qǐng)求;2:取消該請(qǐng)求之前的相同請(qǐng)求
取消該請(qǐng)求之后的相同請(qǐng)求:
在使用 axios 發(fā)送請(qǐng)求時(shí),如果在短時(shí)間內(nèi)連續(xù)發(fā)送同一個(gè)請(qǐng)求,可能會(huì)出現(xiàn)請(qǐng)求結(jié)果混亂或重復(fù)響應(yīng)的問題。為了避免這種情況,我們可以使用 axios 的取消請(qǐng)求功能,每次發(fā)送請(qǐng)求前先取消之前的請(qǐng)求。下面是使用 axios 取消請(qǐng)求的示例代碼:
import axios from 'axios'; // 創(chuàng)建一個(gè) CancelToken 對(duì)象 const source = axios.CancelToken.source(); // 發(fā)送請(qǐng)求 axios.get('/api/data', { // 將 cancelToken 屬性設(shè)置為上面創(chuàng)建的 CancelToken 實(shí)例 cancelToken: source.token }).then(response => { console.log(response.data); }).catch(error => { console.error(error); }); // 取消之前的請(qǐng)求 source.cancel('Duplicate request');
在上面的代碼中,我們先創(chuàng)建一個(gè) CancelToken 對(duì)象 source ,并將其傳遞給發(fā)送請(qǐng)求的 cancelToken 屬性中。然后在需要取消之前請(qǐng)求的時(shí)候,調(diào)用 source.cancel() 方法即可。此時(shí),后續(xù)的請(qǐng)求將不會(huì)被發(fā)送,以避免重復(fù)響應(yīng)。
需要注意的是,如果在請(qǐng)求被取消之前,請(qǐng)求已經(jīng)被發(fā)送并成功響應(yīng),那么該已經(jīng)成功響應(yīng)的請(qǐng)求結(jié)果無法被取消,因此也需要對(duì)已經(jīng)成功響應(yīng)的請(qǐng)求結(jié)果做好處理。另外,也需要處理請(qǐng)求被取消的異常情況,以避免出現(xiàn)錯(cuò)誤。
取消該請(qǐng)求之前的相同請(qǐng)求:
在 axios 中,取消上一次請(qǐng)求與取消前面的所有請(qǐng)求是有差別的。取消上一次請(qǐng)求,我們可以記錄當(dāng)前請(qǐng)求的 cancelToken ,然后在下一次請(qǐng)求之前發(fā)送一個(gè)取消請(qǐng)求的操作。而取消前面的所有請(qǐng)求,我們需要維護(hù)一份請(qǐng)求隊(duì)列,然后在新的請(qǐng)求發(fā)起之前,將之前的所有請(qǐng)求都取消掉。
下面是一個(gè)實(shí)現(xiàn)取消前面的所有請(qǐng)求的例子:
import axios from 'axios'; // 創(chuàng)建一個(gè)請(qǐng)求隊(duì)列 let pending = []; const CancelToken = axios.CancelToken; const removePending = (config) => { for (let p in pending) { if (pending[p].url === config.url + '&' + config.method) { pending[p].cancel("request canceled"); pending.splice(p, 1); } } } // 添加請(qǐng)求攔截器 axios.interceptors.request.use( (config) => { removePending(config); config.cancelToken = new CancelToken((c) => { pending.push({ cancel: c, url: config.url + '&' + config.method }); }); return config; }, (error) => { return Promise.reject(error); } ); // 響應(yīng)攔截器 axios.interceptors.response.use( (response) => { removePending(response.config); return response; }, (error) => { return Promise.reject(error); } );
在上面的代碼中,我們創(chuàng)建了一個(gè)請(qǐng)求隊(duì)列 pending ,然后為請(qǐng)求配置添加了一個(gè) cancelToken ,并將其推入請(qǐng)求隊(duì)列中。在發(fā)送新的請(qǐng)求之前,我們遍歷請(qǐng)求隊(duì)列,將之前的所有請(qǐng)求都取消掉,并從請(qǐng)求隊(duì)列中移除。同時(shí),我們還需要在響應(yīng)攔截器中將已經(jīng)完成的請(qǐng)求從請(qǐng)求隊(duì)列中移除。
需要注意的是,如果發(fā)送的請(qǐng)求是無法取消的,例如使用 jsonp 發(fā)送的請(qǐng)求,那么上面的代碼將無法取消這類請(qǐng)求。因此,我們需要根據(jù)實(shí)際情況選擇合適的方式來取消請(qǐng)求。
取消之后再次發(fā)送請(qǐng)求提示被cancel怎么辦?
如果在 axios 中取消了一個(gè)請(qǐng)求,那么再次發(fā)送同樣的請(qǐng)求會(huì)提示 Cancellation 已經(jīng)被取消的錯(cuò)誤。這是因?yàn)?axios 同樣的請(qǐng)求已經(jīng)被 cancel ,因此無法再次發(fā)送。如果需要再次發(fā)送同樣的請(qǐng)求,需要重新創(chuàng)建一個(gè)新的 cancelToken ,否則會(huì)一直提示被取消。
下面是一個(gè)重新創(chuàng)建 cancelToken 的示例代碼:
import axios from 'axios'; let cancelToken = axios.CancelToken; let cancel; let config = { method: 'get', url: '/api/data', cancelToken: new cancelToken(function executor(c) { // executor 函數(shù)接收一個(gè) cancel 函數(shù)作為參數(shù) cancel = c; }) }; // 添加請(qǐng)求攔截器 axios.interceptors.request.use( (config) => { // 如果 cancel 存在說明上次請(qǐng)求被取消了,重新創(chuàng)建一個(gè) if (typeof cancel === 'function') { config.cancelToken = new cancelToken(function executor(c) { cancel = c; }); } return config; }, (error) => { return Promise.reject(error); } ); // 發(fā)送請(qǐng)求 axios(config) .then((response) => { console.log(response); }) .catch((error) => { if (axios.isCancel(error)) { console.log('Request canceled', error.message); } else { console.log(error); } }); // 取消請(qǐng)求 cancel();
在上面的代碼中,我們?cè)诎l(fā)送請(qǐng)求前先創(chuàng)建了一個(gè) cancelToken ,并將 cancel 函數(shù)保存下來。當(dāng)發(fā)送請(qǐng)求時(shí),我們檢查 cancel 函數(shù)是否存在,如果存在說明上次請(qǐng)求被取消了,那么我們需要重新創(chuàng)建一個(gè) cancelToken 并將其傳遞給新的請(qǐng)求中。這樣就可以在取消請(qǐng)求后重新發(fā)送同樣的請(qǐng)求了。
由于axios是對(duì)XHR對(duì)象的封裝,如果請(qǐng)求已經(jīng)被發(fā)送并且服務(wù)器已經(jīng)接收到請(qǐng)求,那么cancelToken 在調(diào)用 abort() 方法取消請(qǐng)求可能會(huì)導(dǎo)致一部分?jǐn)?shù)據(jù)已經(jīng)被傳輸。因此,使用 時(shí)需要謹(jǐn)慎。
fetch請(qǐng)求才可以做到真正意義上的取消!?。?/p>
總結(jié)
到此這篇關(guān)于axios取消請(qǐng)求CancelToken的用法的文章就介紹到這了,更多相關(guān)axios取消請(qǐng)求CancelToken內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
解決vue項(xiàng)目Error:Cannot find module‘xxx’類報(bào)錯(cuò)問題
當(dāng)npm運(yùn)行報(bào)錯(cuò)Error:Cannot find module 'xxx'時(shí),通常是因?yàn)閚ode_modules文件或依賴未正確安裝,解決步驟包括刪除node_modules和package-lock.json文件,重新運(yùn)行npm install,并根據(jù)需要安裝額外插件,若網(wǎng)絡(luò)問題導(dǎo)致安裝失敗2024-10-10vue實(shí)現(xiàn)表格動(dòng)態(tài)嵌入折線圖的繪制代碼
這篇文章給大家介紹了vue實(shí)現(xiàn)表格動(dòng)態(tài)嵌入折線圖的繪制方法,文中有詳細(xì)完整的代碼示例攻大家參考,對(duì)大家的學(xué)習(xí)或工作有一定的參考價(jià)值,需要的朋友可以參考下2023-10-10vue項(xiàng)目中自定義video視頻控制條的實(shí)現(xiàn)代碼
這篇文章主要介紹了vue項(xiàng)目中自定義video視頻控制條的實(shí)現(xiàn)代碼,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-04-04vue?element-plus中el-input修改邊框border的方法
element樣式還是蠻好的,只是有時(shí)候我們需要做一些調(diào)整,比如el-input的邊框,下面這篇文章主要給大家介紹了關(guān)于vue?element-plus中el-input修改邊框border的相關(guān)資料,需要的朋友可以參考下2022-09-09Vue3 setup語法糖銷毀一個(gè)或多個(gè)定時(shí)器(setTimeout/setInterval)
這篇文章主要給大家介紹了關(guān)于Vue3 setup語法糖銷毀一個(gè)或多個(gè)定時(shí)器(setTimeout/setInterval)的相關(guān)資料,vue是單頁面應(yīng)用,路由切換后,定時(shí)器并不會(huì)自動(dòng)關(guān)閉,需要手動(dòng)清除,當(dāng)頁面被銷毀時(shí),清除定時(shí)器即可,需要的朋友可以參考下2023-10-10vue3?父控件遠(yuǎn)程獲取數(shù)據(jù)在子組件上顯示不出來的解決方案
這篇文章主要介紹了vue3?父控件遠(yuǎn)程獲取數(shù)據(jù),在子組件上顯示不出來,本文給大家分享兩種解決方案幫助大家解決這個(gè)問題,需要的朋友可以參考下2023-08-08在vue中axios設(shè)置timeout超時(shí)的操作
這篇文章主要介紹了在vue中axios設(shè)置timeout超時(shí)的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-09-09詳解vite+ts快速搭建vue3項(xiàng)目以及介紹相關(guān)特性
這篇文章主要介紹了vite+ts快速搭建vue3項(xiàng)目以及介紹相關(guān)特性,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-02-02