亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > JavaScript > 正文

詳解Vue-Router源碼分析路由實現原理

2019-11-19 11:33:27
字體:
來源:轉載
供稿:網友

深入Vue-Router源碼分析路由實現原理

使用Vue開發SPA應用,離不開vue-router,那么vue和vue-router是如何協作運行的呢,下面從使用的角度,大白話幫大家一步步梳理下vue-router的整個實現流程。

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

一、vue-router 源碼結構

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

components下是兩個組件<router-view> 和 <router-link>

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

util下主要是各種功能類和功能函數

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

index是VueRouter類,也整個插件的入口

Install 提供安裝的方法

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

import Vue from 'vue'import VueRouter from 'vue-router'//注冊插件 如果是在瀏覽器環境運行的,可以不寫該方法Vue.use(VueRouter)// 1. 定義(路由)組件。// 可以從其他文件 import 進來const User = { template: '<div>用戶</div>' }const Role = { template: '<div>角色</div>' }// 2. 定義路由// Array,每個路由應該映射一個組件。const routes = [ { path: '/user', component: User }, { path: '/home', component: Home }]// 3. 創建 router 實例,并傳 `routes` 配置const router = new VueRouter({ routes })// 4. 創建和掛載根實例。// 記得要通過 router 對象以參數注入Vue,// 從而讓整個應用都有路由功能// 使用 router-link 組件來導航.// 路由出口// 路由匹配到的組件將渲染在這里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方法會檢測注入插件VueRouter內的install方法,如果有,則執行install方法。
注意:如果是在瀏覽器環境,在index.js內會自動調用.use方法。如果是基于node環境,需要手動調用。

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

Install解析 (對應目錄結構的install.js)

該方法內主要做了以下三件事:

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

 

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({  //對Vue實例混入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)  } }) //通過Vue.prototype定義$router、$route 屬性(方便所有組件可以獲取這兩個屬性) Object.defineProperty(Vue.prototype, '$router', {  get () { return this._routerRoot._router } }) Object.defineProperty(Vue.prototype, '$route', {  get () { return this._routerRoot._route } }) //Vue上注冊router-link和router-view兩個組件 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實例

const router = new VueRouter({ routes })

生成實例過程中,主要做了以下兩件事

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

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

第三步 生成vue實例

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的生命周期,還記得第一步Vue-Router對Vue混入了beforeCreate鉤子嗎,在此會執行哦

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

代碼執行到這,初始化結束,界面將顯示默認首頁

路由更新方式:

一、主動觸發

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

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

HashHistory和HTML5History會分別監控hashchange和popstate來對路由變化作對用的處理 。
HashHistory和HTML5History捕獲到變化后會對應執行push或replace方法,從而調用transitionTo
,剩下的就和上面主動觸發一樣啦。

總結

 1、安裝插件

混入beforeCreate生命周期處理,初始化_routerRoot,_router,_route等數據
全局設置vue靜態訪問router和router和route,方便后期訪問
完成了router-link和 router-view 兩個組件的注冊,router-link用于觸發路由的變化,router-view作 為功能組件,用于觸發對應路由視圖的變化

2、根據路由配置生成router實例

根據配置數組生成路由配置記錄表
生成監控路由變化的hsitory對象

3、將router實例傳入根vue實例

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

router-link綁定了click方法,觸發history.push或者history.replace,從而觸發history.transitionTo
同時會監控hashchange和popstate來對路由變化作對用的處理

以上所述是小編給大家介紹的Vue-Router源碼分析路由實現原理詳解整合,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對武林網網站的支持!

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
高清亚洲成在人网站天堂| 欧亚精品中文字幕| 中文字幕亚洲在线| 精品国产福利在线| 国产精品久久激情| 欧美国产日韩免费| 欧美综合在线第二页| 九色精品免费永久在线| 亚洲国产精品国自产拍av秋霞| 久久久女女女女999久久| 91精品国产91| 日韩va亚洲va欧洲va国产| 91精品在线国产| 国精产品一区一区三区有限在线| 2019精品视频| 成人精品一区二区三区电影免费| 国产性猛交xxxx免费看久久| 久久91精品国产| 亚洲综合精品一区二区| 亚洲国模精品私拍| 91精品综合久久久久久五月天| 2018中文字幕一区二区三区| 亚洲精品美女久久久| 激情成人在线视频| 亚洲女人被黑人巨大进入| 日韩在线观看免费高清完整版| 狠狠躁夜夜躁人人爽天天天天97| 日韩一区二区在线视频| 国产美女91呻吟求| 欧美电影在线观看完整版| 91国自产精品中文字幕亚洲| 成人有码视频在线播放| 北条麻妃在线一区二区| 伊人久久大香线蕉av一区二区| 91精品在线播放| 91视频国产高清| 亚洲老头老太hd| 成人淫片在线看| 久久香蕉国产线看观看av| 欧美激情极品视频| 51精品国产黑色丝袜高跟鞋| 欧美性xxxxxxxxx| 亚洲精品国产综合久久| 日韩欧美在线字幕| 成人97在线观看视频| 欧美国产日韩视频| 国内精品久久久久| 伊人久久免费视频| 69久久夜色精品国产69| 国产美女主播一区| 日韩精品一二三四区| 亚洲天堂成人在线视频| 欧美中文字幕精品| 亚洲国产精彩中文乱码av| 日韩理论片久久| 亚洲黄色在线看| 日本亚洲欧美成人| 成人免费高清完整版在线观看| 亚洲一区二区中文字幕| 欧美高清在线视频观看不卡| 久久精品国产v日韩v亚洲| 亚洲人成网站在线播| 久久久日本电影| 日韩一区二区在线视频| 亚洲第一精品福利| 国产一区二区丝袜| 日韩视频免费大全中文字幕| 人人爽久久涩噜噜噜网站| 久久久噜噜噜久久久| 91av在线播放视频| 欧美亚洲在线观看| 国产精品久久久久久久久久ktv| 国产精品com| 亚洲free嫩bbb| 亚洲男人天堂网| 78色国产精品| 欧美激情视频免费观看| 日韩精品高清在线| 亚洲免费福利视频| 国产成人精品久久二区二区91| 欧美在线视频免费观看| 久久中国妇女中文字幕| 国产欧美va欧美va香蕉在| 国产精品久久久久久久久免费| 欧美激情国产高清| 日本久久久a级免费| 91福利视频在线观看| 国产激情999| 97精品视频在线观看| 久久99国产精品自在自在app| 国产精品美女久久久久久免费| 久久精品电影一区二区| 亚洲人成网在线播放| 欧美视频不卡中文| 日本久久久久久久久久久| 亚洲最大福利网站| 久久九九国产精品怡红院| 国产亚洲精品日韩| 久久久国产精彩视频美女艺术照福利| 久久人人爽人人爽人人片av高清| 精品国内亚洲在观看18黄| 久久久成人av| 亚洲a在线播放| 国产精品成人一区二区三区吃奶| 国内精品400部情侣激情| 成人黄在线观看| 亚洲精品98久久久久久中文字幕| 欧美亚洲在线观看| 91精品国产电影| 久久久亚洲网站| 成人黄色免费片| 日韩在线播放一区| 欧美疯狂做受xxxx高潮| 亚洲欧洲在线看| 欧美成人亚洲成人| 亲子乱一区二区三区电影| 亚洲经典中文字幕| 日日狠狠久久偷偷四色综合免费| 欧美成年人视频网站| 国产欧美日韩精品丝袜高跟鞋| 成人激情视频在线播放| 中文.日本.精品| 国产精品免费视频久久久| 久久久久久成人精品| 色老头一区二区三区在线观看| 日韩中文字幕在线| 国产精品永久在线| 国产午夜精品全部视频播放| 91久久久久久久久久| 精品国产鲁一鲁一区二区张丽| 91精品国产91久久久久久久久| 国产精品99蜜臀久久不卡二区| 97在线视频一区| 亚洲第一精品久久忘忧草社区| 国产精品白丝av嫩草影院| 亚洲图片欧洲图片av| 97精品国产91久久久久久| 国产丝袜精品视频| 成人免费大片黄在线播放| 久久久国产精品x99av| 国产精品久久综合av爱欲tv| 久久久精品国产一区二区| 亚洲人成绝费网站色www| 久久免费视频在线观看| 精品久久久一区| 久久久噜噜噜久久久| 一本色道久久88综合日韩精品| 最近中文字幕mv在线一区二区三区四区| 欧美孕妇毛茸茸xxxx| 国产日本欧美一区二区三区在线| 国产精品久久综合av爱欲tv| 国产精品视频公开费视频| 亚洲2020天天堂在线观看| 91视频免费网站| 欧美最顶级丰满的aⅴ艳星| 亚洲成人精品视频| 美女啪啪无遮挡免费久久网站| 啪一啪鲁一鲁2019在线视频| 欧美成人免费在线视频| 精品偷拍一区二区三区在线看| 一区二区国产精品视频| 欧美黑人巨大精品一区二区| 91精品久久久久久久久久久| 亚洲男女性事视频|