vue-router鉤子執(zhí)行順序示例解析
Vue路由執(zhí)行跳轉(zhuǎn)
Vue的路由在執(zhí)行跳轉(zhuǎn)時,根據(jù)源碼可知,調(diào)用了router中定義的navigate函數(shù)
function push(to: RouteLocationRaw) { return pushWithRedirect(to) } function replace(to: RouteLocationRaw) { return push(assign(locationAsObject(to), { replace: true })) } function pushWithRedirect( to: RouteLocationRaw | RouteLocation, redirectedFrom?: RouteLocation ): Promise<NavigationFailure | void | undefined> { // ... return (failure ? Promise.resolve(failure) : navigate(toLocation, from))/*調(diào)用navigate*/ .catch((error: NavigationFailure | NavigationRedirectError) => isNavigationFailure(error) ? // navigation redirects still mark the router as ready isNavigationFailure(error, ErrorTypes.NAVIGATION_GUARD_REDIRECT) ? error : markAsReady(error) // also returns the error : // reject any unknown error triggerError(error, toLocation, from) ) .then((failure: NavigationFailure | NavigationRedirectError | void) => { if (failure) { // ... } else { // 執(zhí)行finalizeNavigation完成導(dǎo)航 // if we fail we don't finalize the navigation failure = finalizeNavigation( toLocation as RouteLocationNormalizedLoaded, from, true, replace, data ) } // 觸發(fā)`afterEach` triggerAfterEach( toLocation as RouteLocationNormalizedLoaded, from, failure ) return failure }) } function navigate( to: RouteLocationNormalized, from: RouteLocationNormalizedLoaded ): Promise<any> { let guards: Lazy<any>[] // ... // run the queue of per route beforeRouteLeave guards return ( // 1.調(diào)用離開組件的`beforeRouteLeave`鉤子 runGuardQueue(guards) .then(() => { // 獲取全局的的`beforeEach`鉤子 // check global guards beforeEach guards = [] for (const guard of beforeGuards.list()) { guards.push(guardToPromiseFn(guard, to, from)) } guards.push(canceledNavigationCheck) // 2.調(diào)用全局的`beforeEach`鉤子 return runGuardQueue(guards) }) .then(() => { // 獲取更新的路由其對應(yīng)組件的`beforeRouteUpdate`鉤子 // check in components beforeRouteUpdate guards = extractComponentsGuards( updatingRecords, 'beforeRouteUpdate', to, from ) for (const record of updatingRecords) { record.updateGuards.forEach(guard => { guards.push(guardToPromiseFn(guard, to, from)) }) } guards.push(canceledNavigationCheck) // 3.調(diào)用復(fù)用組件的`beforeRouteUpdate`鉤子 // run the queue of per route beforeEnter guards return runGuardQueue(guards) }) .then(() => { // 獲取進(jìn)入的路由自身的`beforeEnter`鉤子 // check the route beforeEnter guards = [] for (const record of enteringRecords) { // do not trigger beforeEnter on reused views if (record.beforeEnter) { if (isArray(record.beforeEnter)) { for (const beforeEnter of record.beforeEnter) guards.push(guardToPromiseFn(beforeEnter, to, from)) } else { guards.push(guardToPromiseFn(record.beforeEnter, to, from)) } } } guards.push(canceledNavigationCheck) // 4.調(diào)用新匹配路由的`beforeEnter`鉤子 // run the queue of per route beforeEnter guards return runGuardQueue(guards) }) .then(() => { // NOTE: at this point to.matched is normalized and does not contain any () => Promise<Component> // clear existing enterCallbacks, these are added by extractComponentsGuards to.matched.forEach(record => (record.enterCallbacks = {})) // 獲取進(jìn)入的路由其對應(yīng)組件的`beforeRouteEnter`鉤子 // check in-component beforeRouteEnter guards = extractComponentsGuards( enteringRecords, 'beforeRouteEnter', to, from ) guards.push(canceledNavigationCheck) // 5.調(diào)用新組件的`beforeRouteEnter`鉤子 // run the queue of per route beforeEnter guards return runGuardQueue(guards) }) .then(() => { // 獲取全局的的`beforeResolve`鉤子 // check global guards beforeResolve guards = [] for (const guard of beforeResolveGuards.list()) { guards.push(guardToPromiseFn(guard, to, from)) } guards.push(canceledNavigationCheck) // 6.調(diào)用全局的`beforeResolve`守衛(wèi) return runGuardQueue(guards) }) // catch any navigation canceled .catch(err => isNavigationFailure(err, ErrorTypes.NAVIGATION_CANCELLED) ? err : Promise.reject(err) ) ) } // 觸發(fā)`afterEach` function triggerAfterEach( to: RouteLocationNormalizedLoaded, from: RouteLocationNormalizedLoaded, failure?: NavigationFailure | void ): void { // navigation is confirmed, call afterGuards // TODO: wrap with error handlers afterGuards .list() .forEach(guard => runWithContext(() => guard(to, from, failure))) }
順序執(zhí)行
由上述源碼中可以看出,由Promise then的鏈?zhǔn)秸{(diào)用保證了路由守衛(wèi)按照以下順序執(zhí)行:
- 舊的路由組件
beforeRouteLeave
- 全局配置的
beforeEach
- 復(fù)用的路由組件
beforeRouteUpdate
- 新路由的
beforeEnter
- 新路由組件的
beforeRouteEnter
- 全局配置的
beforeResolve
- navigate執(zhí)行完畢后,會調(diào)用
triggerAfterEach
函數(shù),觸發(fā)afterEach
和官網(wǎng)文檔上所寫的是一樣的。The Full Navigation Resolution Flow
以上就是vue-router鉤子執(zhí)行順序的詳細(xì)內(nèi)容,更多關(guān)于vue-router鉤子執(zhí)行順序的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
element多選表格中使用Switch開關(guān)的實現(xiàn)
當(dāng)在做后臺管理系統(tǒng)的時候,會用到用戶的狀態(tài)管理這個功能,本文主要介紹了element多選表格中使用Switch開關(guān)的實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-07-07vue菜單欄聯(lián)動內(nèi)容頁面tab的實現(xiàn)示例
本文主要介紹了vue菜單欄聯(lián)動內(nèi)容頁面tab的實現(xiàn)示例,左側(cè)菜單欄與右側(cè)內(nèi)容部分聯(lián)動,當(dāng)點擊左側(cè)的菜單,右側(cè)會展示對應(yīng)的tab,具有一定的參考價值,感興趣的可以了解一下2024-01-01Vue使用pinia管理數(shù)據(jù)pinia持久化存儲問題
這篇文章主要介紹了Vue使用pinia管理數(shù)據(jù)pinia持久化存儲問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-03-03Vue2.0父組件與子組件之間的事件發(fā)射與接收實例代碼
這篇文章主要介紹了Vue2.0父組件與子組件之間的事件發(fā)射與接收實例代碼,需要的朋友可以參考下2017-09-09Vue el-table表頭上引入組件不能實時傳參解決方法分析
這篇文章主要介紹了Vue el-table表頭上引入組件不能實時傳參解決方法,總的來說這并不是一道難題,那為什么要拿出這道題介紹?拿出這道題真正想要傳達(dá)的是解題的思路,以及不斷優(yōu)化探尋最優(yōu)解的過程。希望通過這道題能給你帶來一種解題優(yōu)化的思路2022-11-11解決Vue使用百度地圖BMapGL內(nèi)存泄漏問題?Out?of?Memory
這篇文章主要介紹了解決Vue使用百度地圖BMapGL內(nèi)存泄漏問題?Out?of?Memory,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-12-12vue實現(xiàn)頁面上傳文件夾壓縮后傳給服務(wù)器的操作
這篇文章主要介紹了vue實現(xiàn)頁面上傳文件夾壓縮后傳給服務(wù)器的操作,本文通過示例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-09-09Vue中實現(xiàn)父子組件雙向數(shù)據(jù)流的三種方案分享
通常情況下,父子組件的通信都是單向的,或父組件使用props向子組件傳遞數(shù)據(jù),或子組件使用emit函數(shù)向父組件傳遞數(shù)據(jù),本文將嘗試講解Vue中常用的幾種雙向數(shù)據(jù)流的使用,需要的朋友可以參考下2023-08-08