vue-router鉤子執(zhí)行順序示例解析
更新時間:2023年07月28日 08:53:15 作者:beckyyyy
這篇文章主要為大家介紹了vue-router鉤子執(zhí)行順序示例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
Vue路由執(zhí)行跳轉
Vue的路由在執(zhí)行跳轉時,根據源碼可知,調用了router中定義的navigate函數
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))/*調用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完成導航 // 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.調用離開組件的`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.調用全局的`beforeEach`鉤子 return runGuardQueue(guards) }) .then(() => { // 獲取更新的路由其對應組件的`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.調用復用組件的`beforeRouteUpdate`鉤子 // run the queue of per route beforeEnter guards return runGuardQueue(guards) }) .then(() => { // 獲取進入的路由自身的`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.調用新匹配路由的`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 = {})) // 獲取進入的路由其對應組件的`beforeRouteEnter`鉤子 // check in-component beforeRouteEnter guards = extractComponentsGuards( enteringRecords, 'beforeRouteEnter', to, from ) guards.push(canceledNavigationCheck) // 5.調用新組件的`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.調用全局的`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的鏈式調用保證了路由守衛(wèi)按照以下順序執(zhí)行:
- 舊的路由組件
beforeRouteLeave
- 全局配置的
beforeEach
- 復用的路由組件
beforeRouteUpdate
- 新路由的
beforeEnter
- 新路由組件的
beforeRouteEnter
- 全局配置的
beforeResolve
- navigate執(zhí)行完畢后,會調用
triggerAfterEach
函數,觸發(fā)afterEach
和官網文檔上所寫的是一樣的。The Full Navigation Resolution Flow
以上就是vue-router鉤子執(zhí)行順序的詳細內容,更多關于vue-router鉤子執(zhí)行順序的資料請關注腳本之家其它相關文章!
相關文章
Vue2.0父組件與子組件之間的事件發(fā)射與接收實例代碼
這篇文章主要介紹了Vue2.0父組件與子組件之間的事件發(fā)射與接收實例代碼,需要的朋友可以參考下2017-09-09Vue el-table表頭上引入組件不能實時傳參解決方法分析
這篇文章主要介紹了Vue el-table表頭上引入組件不能實時傳參解決方法,總的來說這并不是一道難題,那為什么要拿出這道題介紹?拿出這道題真正想要傳達的是解題的思路,以及不斷優(yōu)化探尋最優(yōu)解的過程。希望通過這道題能給你帶來一種解題優(yōu)化的思路2022-11-11解決Vue使用百度地圖BMapGL內存泄漏問題?Out?of?Memory
這篇文章主要介紹了解決Vue使用百度地圖BMapGL內存泄漏問題?Out?of?Memory,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-12-12