詳解Axios統(tǒng)一錯(cuò)誤處理與后置
問題
在進(jìn)行業(yè)務(wù)開發(fā)的時(shí)候,前后端會對接口的數(shù)據(jù)結(jié)構(gòu)進(jìn)行約定,若接口有異常,需要將異常信息展示給用戶知曉。這個(gè)流程里,數(shù)據(jù)結(jié)構(gòu)是確定的(事先約定),數(shù)據(jù)的處理邏輯是相同的(展示給用戶),如果在業(yè)務(wù)代碼代碼中重復(fù)的catch(e) { 展示給用戶 },就非常的不優(yōu)雅。本著Don't repeat myself(懶)的原則,需要對接口錯(cuò)誤進(jìn)行統(tǒng)一處理。
接下來,我會結(jié)合具體的業(yè)務(wù)場景,講一講我的解決方案。
業(yè)務(wù)場景
- 后端通過http狀態(tài)標(biāo)識接口狀態(tài),錯(cuò)誤信息在response的data里
- 前端的處理邏輯是使用element-ui的Message展示錯(cuò)誤信息
- 使用axios
axios可以通過攔截器,在業(yè)務(wù)代碼處理響應(yīng)之前對響應(yīng)進(jìn)行處理,類似于下面的流程
someAPI() .then(interceptorsFn) .then(業(yè)務(wù)邏輯)
所以,我們可以在interceptors對響應(yīng)進(jìn)行統(tǒng)一處理:
request.interceptors.response.use(
(response) => response.data,
(error) => {
// 針對特定的http狀態(tài)碼進(jìn)行處理
if (error.response && error.response.status === 401) {
router.push({ name: 'ssoLogin' })
return new Promise(() => {}) // pending的promise,中止promise鏈
}
.....
const msg = error.response.data
Message.error(msg)
return Promise.reject(error.response)
}
)
如何進(jìn)行特定的錯(cuò)誤處理
不難看出,上面的方案有一個(gè)問題,如果有某個(gè)接口需要有業(yè)務(wù)代碼來展示定制的錯(cuò)誤信息(這個(gè)情況十分常見),如何處理?
naive方案1:業(yè)務(wù)代碼使用其它的方式展示信息:例如Notify。
這個(gè)方案被我司產(chǎn)品痛罵,因?yàn)槠茐牧私y(tǒng)一的錯(cuò)誤信息展示,并且此時(shí)統(tǒng)一的錯(cuò)誤信息是一個(gè)垃圾信息,沒必要展示。
naive方案2:業(yè)務(wù)代碼直接使用Message,頂?shù)艚y(tǒng)一的錯(cuò)誤信息。
這個(gè)方案還是被產(chǎn)品大哥(dog)懟了,因?yàn)槊黠@的用戶體驗(yàn)不好,錯(cuò)誤信息出現(xiàn)了閃爍。
帥氣的解決方案3:業(yè)務(wù)代碼決定是否隱藏統(tǒng)一錯(cuò)誤提示
那么問題來了,由于是先走攔截器,再走業(yè)務(wù)代碼,如何由業(yè)務(wù)代碼決定是否隱藏統(tǒng)一錯(cuò)誤提示呢?
我的辦法是,將統(tǒng)一的錯(cuò)誤提示使用setTimeout放到下一個(gè)loop執(zhí)行,并通過一個(gè)變量標(biāo)識是否要執(zhí)行統(tǒng)一錯(cuò)誤提示。
request.interceptors.response.use(
(response) => response.data,
(error) => {
...
setTimeout(() => {
if (tag) {
Message.error(msg)
}
})
}
)
接下來,需要考慮的是,如何在業(yè)務(wù)代碼里改變標(biāo)識變量
naive方案1:一個(gè)全局的變量或者方法
這個(gè)方案非常的不靠譜,若在其它代碼里改變了這個(gè)全局變量,就嗝屁,并且N個(gè)接口公用一個(gè)標(biāo)識變量,只能是同一個(gè)狀態(tài)。
帥氣方案2:
request.interceptors.response.use(
(response) => response.data,
(error) => {
...
let isShowNormalError = true
const hideNormalError = () => isShowNormalError = false
setTimeout(() => {
if (isShowNormalError) {
Message.error(msg)
}
})
return Promise.reject({ ...error.response, hideNormalMessage }) // 在error.response上添加方法
}
)
業(yè)務(wù)代碼:
someAPIFN()
.then()
.catch({ data, hideNormalMessage }) {
// 業(yè)務(wù)代碼
hideNormalMessage()
}
兼容舊代碼
目前的方案需要對現(xiàn)存代碼做修改,對進(jìn)行特殊處理的接口添加hideNormalMessage()。如果不想全局搜索添加代碼(懶),可以根據(jù)業(yè)務(wù)來進(jìn)行兼容。下面講一下我結(jié)合業(yè)務(wù)代碼進(jìn)行的兼容處理(非常不推薦)。
request.interceptors.response.use(
(response) => response.data,
(error) => {
// warning,和業(yè)務(wù)代碼深度耦合,不推薦
const hasMessageBeforeCatch = !!document.querySelector('.el-message')
...
let isShowNormalError = true
const hideNormalError = () => isShowNormalError = false
setTimeout(() => {
const hasMessageAfterCatch = document.querySelector('.el-message')
// 調(diào)用catch前沒有message,調(diào)用catch后有message,表示message是在catch過程中產(chǎn)生
const madeMessageWhenCatch = !hasMessageBeforeCatch && hasMessageAfterCatch
if (isShowNormalError && !madeMessageWhenCatch) {
Message.error(msg)
}
})
return Promise.reject({ ...error.response, hideNormalMessage }) // 在error.response上添加方法
}
)
邏輯:如果在catch中使用了Message,就不展示統(tǒng)一錯(cuò)誤處理
總結(jié)
這個(gè)解決方案的關(guān)鍵在于使用setTimeout使得統(tǒng)一錯(cuò)誤處理“落后”于業(yè)務(wù)代碼,并在Promise.reject的參數(shù)中添加控制函數(shù)使得業(yè)務(wù)代碼可以決定是否展示統(tǒng)一錯(cuò)誤處理。稍作抽象與封裝就可以形成一個(gè)業(yè)務(wù)無關(guān)、框架無關(guān)的統(tǒng)一錯(cuò)誤處理方案。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Vue組件內(nèi)部實(shí)現(xiàn)一個(gè)雙向數(shù)據(jù)綁定的實(shí)例代碼
這篇文章主要介紹了Vue組件內(nèi)部實(shí)現(xiàn)一個(gè)雙向數(shù)據(jù)綁定的實(shí)例代碼,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-04-04
Vue2.0利用 v-model 實(shí)現(xiàn)組件props雙向綁定的優(yōu)美解決方案
本篇文章主要介紹了Vue2 利用 v-model 實(shí)現(xiàn)組件props雙向綁定的優(yōu)美解決方案,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下。2017-03-03
vue使用CSS插件scss時(shí)代碼報(bào)紅問題
這篇文章主要介紹了vue使用CSS插件scss時(shí)代碼報(bào)紅問題,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-07-07
Vue中 key keep-alive的實(shí)現(xiàn)原理
這篇文章主要介紹了Vue中 key keep-alive的實(shí)現(xiàn)原理,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-09-09
Vue3?Axios攔截器封裝成request文件的示例詳解
這篇文章主要介紹了Vue3?Axios攔截器封裝成request文件,本文通過示例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-04-04
vue3-print-nb實(shí)現(xiàn)頁面打印(含分頁打印)示例代碼
大多數(shù)后臺系統(tǒng)中都存在打印的需求,在有打印需求時(shí),對前端來說當(dāng)然是直接打印頁面更容易,下面這篇文章主要給大家介紹了關(guān)于vue3-print-nb實(shí)現(xiàn)頁面打印(含分頁打印)的相關(guān)資料,需要的朋友可以參考下2024-01-01
Webpack+Vue如何導(dǎo)入Jquery和Jquery的第三方插件
本文主要介紹了Webpack+Vue導(dǎo)入Jquery和Jquery的第三方插件的方法,具有很好的參考價(jià)值,下面跟著小編一起來看下吧2017-02-02
vue3 學(xué)習(xí)筆記之a(chǎn)xios的使用變化總結(jié)
本篇文章主要旨在幫助正在學(xué)vue3或者準(zhǔn)備學(xué)vue3的同學(xué)了解網(wǎng)絡(luò)請求axios該如何使用,防止接觸了一點(diǎn)點(diǎn)vue3的同學(xué)會有個(gè)疑問。有興趣的小伙伴可以關(guān)注一下2021-11-11
vue引入jquery時(shí)報(bào)錯(cuò) $ is not defined的問題及解決
這篇文章主要介紹了vue引入jquery時(shí)報(bào)錯(cuò) $ is not defined的問題及解決,具有很好的參考價(jià)值,希望對大家有所幫助。2022-09-09

