亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

詳解Vue-Router源碼分析路由實(shí)現(xiàn)原理

 更新時(shí)間:2019年05月15日 17:32:13   作者:隕落的軌跡  
這篇文章主要介紹了Vue-Router源碼分析路由實(shí)現(xiàn)原理,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧

深入Vue-Router源碼分析路由實(shí)現(xiàn)原理

使用Vue開發(fā)SPA應(yīng)用,離不開vue-router,那么vue和vue-router是如何協(xié)作運(yùn)行的呢,下面從使用的角度,大白話幫大家一步步梳理下vue-router的整個(gè)實(shí)現(xiàn)流程。

到發(fā)文時(shí)使用的版本是:
- vue (v2.5.0)
- vue-router (v3.0.1)

一、vue-router 源碼結(jié)構(gòu)

github 地址:https://github.com/vuejs/vue-router

components下是兩個(gè)組件<router-view> 和 <router-link>

history是路由方式的封裝,提供三種方式

util下主要是各種功能類和功能函數(shù)

create-matcher和create-router-map是生成匹配表

index是VueRouter類,也整個(gè)插件的入口

Install 提供安裝的方法

先整體展示下vue-router使用方式,請(qǐng)牢記一下幾步哦。

import Vue from 'vue'
import VueRouter from 'vue-router'
//注冊(cè)插件 如果是在瀏覽器環(huán)境運(yùn)行的,可以不寫該方法
Vue.use(VueRouter)

// 1. 定義(路由)組件。
// 可以從其他文件 import 進(jìn)來(lái)
const User = { template: '<div>用戶</div>' }
const Role = { template: '<div>角色</div>' }

// 2. 定義路由
// Array,每個(gè)路由應(yīng)該映射一個(gè)組件。
const routes = [
 { path: '/user', component: User },
 { path: '/home', component: Home }
]

// 3. 創(chuàng)建 router 實(shí)例,并傳 `routes` 配置
const router = new VueRouter({
 routes 
})

// 4. 創(chuàng)建和掛載根實(shí)例。
// 記得要通過(guò) router 對(duì)象以參數(shù)注入Vue,
// 從而讓整個(gè)應(yīng)用都有路由功能
// 使用 router-link 組件來(lái)導(dǎo)航.
// 路由出口
// 路由匹配到的組件將渲染在這里
const app = new Vue({
 router,
 template: `
  <div id="app">
   <h1>Basic</h1>
   <ul>
    <li><router-link to="/">/</router-link></li>
    <li><router-link to="/user">用戶</router-link></li>
    <li><router-link to="/role">角色</router-link></li>
    <router-link tag="li" to="/user">/用戶</router-link>
   </ul>
   <router-view class="view"></router-view>
  </div>
 `
}).$mount('#app')

分析開始

第一步

Vue是使用.use( plugins )方法將插件注入到Vue中。
use方法會(huì)檢測(cè)注入插件VueRouter內(nèi)的install方法,如果有,則執(zhí)行install方法。
注意:如果是在瀏覽器環(huán)境,在index.js內(nèi)會(huì)自動(dòng)調(diào)用.use方法。如果是基于node環(huán)境,需要手動(dòng)調(diào)用。

if (inBrowser && window.Vue) {
 window.Vue.use(VueRouter)
}

Install解析 (對(duì)應(yīng)目錄結(jié)構(gòu)的install.js)

該方法內(nèi)主要做了以下三件事:

  1. 1、對(duì)Vue實(shí)例混入beforeCreate鉤子操作(在Vue的生命周期階段會(huì)被調(diào)用)
  2. 2、通過(guò)Vue.prototype定義router、router、route 屬性(方便所有組件可以獲取這兩個(gè)屬性)
  3. 3、Vue上注冊(cè)router-link和router-view兩個(gè)組件

 

export function install (Vue) {
 if (install.installed && _Vue === Vue) return
 install.installed = true

 _Vue = Vue

 const isDef = v => v !== undefined

 const registerInstance = (vm, callVal) => {
  let i = vm.$options._parentVnode
  if (isDef(i) && isDef(i = i.data) && isDef(i = i.registerRouteInstance)) {
   i(vm, callVal)
  }
 }

 Vue.mixin({
  //對(duì)Vue實(shí)例混入beforeCreate鉤子操作
  beforeCreate () {
   if (isDef(this.$options.router)) {
    this._routerRoot = this
    this._router = this.$options.router
    this._router.init(this)
    Vue.util.defineReactive(this, '_route', this._router.history.current)
   } else {
    this._routerRoot = (this.$parent && this.$parent._routerRoot) || this
   }
   registerInstance(this, this)
  },
  destroyed () {
   registerInstance(this)
  }
 })
 //通過(guò)Vue.prototype定義$router、$route 屬性(方便所有組件可以獲取這兩個(gè)屬性)
 Object.defineProperty(Vue.prototype, '$router', {
  get () { return this._routerRoot._router }
 })

 Object.defineProperty(Vue.prototype, '$route', {
  get () { return this._routerRoot._route }
 })
 //Vue上注冊(cè)router-link和router-view兩個(gè)組件
 Vue.component('RouterView', View)
 Vue.component('RouterLink', Link)

 const strats = Vue.config.optionMergeStrategies
 // use the same hook merging strategy for route hooks
 strats.beforeRouteEnter = strats.beforeRouteLeave = strats.beforeRouteUpdate = strats.created
}

第二步 生成router實(shí)例

const router = new VueRouter({
 routes 
})

生成實(shí)例過(guò)程中,主要做了以下兩件事

  1. 1、根據(jù)配置數(shù)組(傳入的routes)生成路由配置記錄表。
  2. 2、根據(jù)不同模式生成監(jiān)控路由變化的History對(duì)象

注:History類由HTML5History、HashHistory、AbstractHistory三類繼承
history/base.js實(shí)現(xiàn)了基本history的操作
history/hash.js,history/html5.js和history/abstract.js繼承了base,只是根據(jù)不同的模式封裝了一些基本操作

第三步 生成vue實(shí)例

const app = new Vue({
 router,
 template: `
  <div id="app">
   <h1>Basic</h1>
   <ul>
    <li><router-link to="/">/</router-link></li>
    <li><router-link to="/user">用戶</router-link></li>
    <li><router-link to="/role">角色</router-link></li>
    <router-link tag="li" to="/user">/用戶</router-link>
   </ul>
   <router-view class="view"></router-view>
  </div>
 `
}).$mount('#app')

代碼執(zhí)行到這,會(huì)進(jìn)入Vue的生命周期,還記得第一步Vue-Router對(duì)Vue混入了beforeCreate鉤子嗎,在此會(huì)執(zhí)行哦

Vue.mixin({
  beforeCreate () {
   //驗(yàn)證vue是否有router對(duì)象了,如果有,就不再初始化了
   if (isDef(this.$options.router)) { //沒(méi)有router對(duì)象
    //將_routerRoot指向根組件
    this._routerRoot = this
    //將router對(duì)象掛載到根組件元素_router上
    this._router = this.$options.router
    //初始化,建立路由監(jiān)控
    this._router.init(this)
    //劫持?jǐn)?shù)據(jù)_route,一旦_route數(shù)據(jù)發(fā)生變化后,通知router-view執(zhí)行render方法
    Vue.util.defineReactive(this, '_route', this._router.history.current)
   } else {
    //如果有router對(duì)象,去尋找根組件,將_routerRoot執(zhí)行根組件(解決嵌套關(guān)系時(shí)候_routerRoot指向不一致問(wèn)題)
    this._routerRoot = (this.$parent && this.$parent._routerRoot) || this
   }
   registerInstance(this, this)
  },
  destroyed () {
   registerInstance(this)
  }
 })

代碼執(zhí)行到這,初始化結(jié)束,界面將顯示默認(rèn)首頁(yè)

路由更新方式:

一、主動(dòng)觸發(fā)

router-link綁定了click方法,觸發(fā)history.push或者h(yuǎn)istory.replace,從而觸發(fā)history.transitionTo。
transitionTo用于處理路由轉(zhuǎn)換,其中包含了updateRoute用于更新_route。
在beforeCreate中有劫持_route的方法,當(dāng)_route變化后,觸發(fā)router-view的變化。

二、地址變化(如:在瀏覽器地址欄直接輸入地址)

HashHistory和HTML5History會(huì)分別監(jiān)控hashchange和popstate來(lái)對(duì)路由變化作對(duì)用的處理 。
HashHistory和HTML5History捕獲到變化后會(huì)對(duì)應(yīng)執(zhí)行push或replace方法,從而調(diào)用transitionTo
,剩下的就和上面主動(dòng)觸發(fā)一樣啦。

總結(jié)

 1、安裝插件

混入beforeCreate生命周期處理,初始化_routerRoot,_router,_route等數(shù)據(jù)
全局設(shè)置vue靜態(tài)訪問(wèn)router和router和route,方便后期訪問(wèn)
完成了router-link和 router-view 兩個(gè)組件的注冊(cè),router-link用于觸發(fā)路由的變化,router-view作 為功能組件,用于觸發(fā)對(duì)應(yīng)路由視圖的變化

2、根據(jù)路由配置生成router實(shí)例

根據(jù)配置數(shù)組生成路由配置記錄表
生成監(jiān)控路由變化的hsitory對(duì)象

3、將router實(shí)例傳入根vue實(shí)例

根據(jù)beforeCreate混入,為根vue對(duì)象設(shè)置了劫持字段_route,用戶觸發(fā)router-view的變化
調(diào)用init()函數(shù),完成首次路由的渲染,首次渲染的調(diào)用路徑是 調(diào)用history.transitionTo方法,根據(jù)router的match函數(shù),生成一個(gè)新的route對(duì)象
接著通過(guò)confirmTransition對(duì)比一下新生成的route和當(dāng)前的route對(duì)象是否改變,改變的話觸發(fā)updateRoute,更新hsitory.current屬性,觸發(fā)根組件的_route的變化,從而導(dǎo)致組件的調(diào)用render函數(shù),更新router-view
另外一種更新路由的方式是主動(dòng)觸發(fā)

router-link綁定了click方法,觸發(fā)history.push或者h(yuǎn)istory.replace,從而觸發(fā)history.transitionTo
同時(shí)會(huì)監(jiān)控hashchange和popstate來(lái)對(duì)路由變化作對(duì)用的處理

以上所述是小編給大家介紹的Vue-Router源碼分析路由實(shí)現(xiàn)原理詳解整合,希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!

相關(guān)文章

  • Vue下滾動(dòng)到頁(yè)面底部無(wú)限加載數(shù)據(jù)的示例代碼

    Vue下滾動(dòng)到頁(yè)面底部無(wú)限加載數(shù)據(jù)的示例代碼

    本篇文章主要介紹了Vue下滾動(dòng)到頁(yè)面底部無(wú)限加載數(shù)據(jù)的示例代碼,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2018-04-04
  • 關(guān)于vxe-table復(fù)選框翻頁(yè)選中問(wèn)題及解決

    關(guān)于vxe-table復(fù)選框翻頁(yè)選中問(wèn)題及解決

    這篇文章主要介紹了關(guān)于vxe-table復(fù)選框翻頁(yè)選中問(wèn)題及解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-09-09
  • VUE中使用MUI方法

    VUE中使用MUI方法

    在本篇文章里小編給大家分享了關(guān)于VUE中使用MUI方法和步驟,有需要的朋友們可以學(xué)習(xí)參考下。
    2019-02-02
  • vue中可以綁定多個(gè)事件嗎

    vue中可以綁定多個(gè)事件嗎

    這篇文章主要介紹了vue中可以綁定多個(gè)事件嗎,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-10-10
  • vue、element實(shí)現(xiàn)表格表頭縱向顯示方式

    vue、element實(shí)現(xiàn)表格表頭縱向顯示方式

    這篇文章主要介紹了vue、element實(shí)現(xiàn)表格表頭縱向顯示方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-10-10
  • vue項(xiàng)目實(shí)現(xiàn)圖片上傳功能

    vue項(xiàng)目實(shí)現(xiàn)圖片上傳功能

    這篇文章主要為大家詳細(xì)介紹了vue項(xiàng)目實(shí)現(xiàn)圖片上傳功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2019-12-12
  • vue3+elementplus基于el-table-v2封裝公用table組件詳細(xì)代碼

    vue3+elementplus基于el-table-v2封裝公用table組件詳細(xì)代碼

    在日常開發(fā)后端管理系統(tǒng)項(xiàng)目中,用于展示數(shù)據(jù)多會(huì)用表格進(jìn)行展示,下面這篇文章主要給大家介紹了關(guān)于vue3+elementplus基于el-table-v2封裝公用table組件的相關(guān)資料,需要的朋友可以參考下
    2023-12-12
  • vue3 源碼解讀之 time slicing的使用方法

    vue3 源碼解讀之 time slicing的使用方法

    這篇文章主要介紹了vue3 源碼解讀之 time slicing的使用方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2019-10-10
  • vue3如何定義變量及ref、reactive、toRefs特性說(shuō)明

    vue3如何定義變量及ref、reactive、toRefs特性說(shuō)明

    這篇文章主要介紹了vue3如何定義變量及ref、reactive、toRefs特性說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-06-06
  • vue實(shí)現(xiàn)token過(guò)期自動(dòng)跳轉(zhuǎn)到登錄頁(yè)面

    vue實(shí)現(xiàn)token過(guò)期自動(dòng)跳轉(zhuǎn)到登錄頁(yè)面

    本文主要介紹了vue實(shí)現(xiàn)token過(guò)期自動(dòng)跳轉(zhuǎn)到登錄頁(yè)面,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-10-10

最新評(píng)論