學習記錄: 模擬實現(xiàn)vue-router

前置知識

  • 插件
  • 混入
  • Vue.observable()
  • 插槽
  • render函數(shù)
  • 運行時和完整版的vue

首先使用vue-cli創(chuàng)建一個項目
vue create my-vue-router

新增router文件夾,在router文件夾中的index.js添加基礎(chǔ)的路由配置

import Vue from 'vue'
import VueRouter from 'vue-router'
import Home from '../views/Home.vue'
// 注冊插件
Vue.use(VueRouter)

const routes = [
  { path: '/', name: 'Home', component: Home},
  { path: '/about', name: 'About', component: () => import(/* webpackChuckName: "about" */ '../views/About.vue')}
]
// 創(chuàng)建路由對象
const router = new VueRouter({
  mode: 'history',
  base: process.env.BASE_URL,
  routes
})
export default router

接下來我們再來看一個畫好的VueRouter的類圖

image.png
  • options:記錄構(gòu)造函數(shù)中傳入的對象
  • routeMap:用來記錄我們路由地址和組件的對應(yīng)關(guān)系, 我們會把路由規(guī)則解析到routeMap里面來
  • data: 是一個對象,里面有一個current屬性,用來記錄當前路由地址,設(shè)置data的目的是因為我們需要一個響應(yīng)式的對象
  • constructor幫我們初始化上面的幾個屬性
  • init方法用來調(diào)用圖上所示init下面的3個方法
  • initEvent方法用來注冊popstate事件,監(jiān)聽瀏覽器歷史的變化
  • createRouteMap:初始化routeMap屬性,把構(gòu)造函數(shù)中傳入的路由規(guī)則轉(zhuǎn)換成鍵值對的形式存儲到routeMap里面
  • initComponent: 創(chuàng)建router-link和router-view這兩個組件

實現(xiàn)思路

  • 導(dǎo)出一個VueRouter的類,在類里面定義一個install靜態(tài)方法
    1. 判斷當前插件是否已經(jīng)被安裝
    2. 把VUE的構(gòu)造函數(shù)記錄到全局變量中
    3. 把創(chuàng)建Vue實例時傳遞的Router對象注入到所有的vue實例上
let _Vue = null
export default class VueRouter {
  static install (Vue) {
    // 判斷當前插件是否已經(jīng)被安裝
    if (VueRouter.install.installed) {
      return
    }
    VueRouter.install.installed = true

    /*
    *把VUE的構(gòu)造函數(shù)記錄到全局變量中,當前的install方法是一個靜態(tài)方法,
    *在這個靜態(tài)方法中我們接收了一個參數(shù)Vue的構(gòu)造函數(shù),而將來我們在Vue-router中的一些實例方法中還要使用Vue的構(gòu)造函數(shù)
    */
    _Vue = Vue

    // 把創(chuàng)建Vue實例時傳遞的Router對象注入到所有的vue實例上
    // 混入,
    _Vue.mixin({
      beforeCreate() {
        if (this.$options.router) { // 實例的選項才有router,組件的選項沒有router
          _Vue.prototype.$router = this.$options.router
          this.$options.router.init() 
        }
      },
    })
  }
}
  • 創(chuàng)建constructor構(gòu)造函數(shù)
constructor (options) {
    this.options = options // 記錄構(gòu)造函數(shù)中傳入的選項
    this.routeMap = {} // 把options中傳入的路由規(guī)則解析出來,存儲到routeMap, 鍵:儲存的是路由地址,值:存儲的是路由組件;
    this.data = _Vue.observable({ // observable: 創(chuàng)建響應(yīng)式的對象
      current: window.location.pathname // 存儲當前路由地址
    })
  }
  • createRouteMap() 遍歷所有路由規(guī)則
createRouteMap () {
  // 遍歷所有的路由規(guī)則,把路由規(guī)則解析成鍵值對的形式,存儲到routeMap對象中
  this.options.routes.forEach(route => {
    this.routeMap[route.path] = route.component
  })
}
  • 創(chuàng)建initComponent創(chuàng)建組件
initComponent (Vue) {
  Vue.component('router-link', {
    props: {
      to: String
    },
    // 使用運行時版本的
    render (h) {// h的作用是幫我們創(chuàng)建虛擬dom
      // 第一個參數(shù):選擇器,第二個參數(shù):屬性, 事件,第三個參數(shù):子元素
      return h('a', {
        attrs: {
          href: this.to
        },
        on: {
          click: this.clickHandler
        }
      }, [this.$slots.default])
    },
    methods: {
      clickHandler (e) {
        history.pushState({}, '', this.to)
        this.$router.data.current = this.to
        e.preventDefault()
      }
    }
    // 需要帶編譯器版本的
    // template: '<a :href="to"><slot></slot></a>'
  })
  const self = this
  Vue.component('router-view', {
    render (h) {
      const component = self.routeMap[self.data.current]
      return h(component)
    }
  })
}
  • 創(chuàng)建initEvent方法用來注冊popstate事件
initEvent () {
  window.addEventListener('popstate', () => {
    this.data.current = window.location.pathname
  })
}
  • init方法調(diào)用
init () {
  this.createRouteMap()
  this.initComponent(_Vue)
  this.initEvent()
}

完整代碼

let _Vue = null
export default class VueRouter {
  // Vue.use()中調(diào)用install方法的時候會傳遞兩個參數(shù), 一個是VUE的構(gòu)造函數(shù),第二個參數(shù)是可選的選項對象
  /**
   * 判斷當前插件是否已經(jīng)被安裝,如果已經(jīng)安裝了就不需要重復(fù)安裝
   * 把VUE的構(gòu)造函數(shù)記錄到全局變量中來,當前的install方法是一個靜態(tài)方法,在這個靜態(tài)方法中我們接收了一個參數(shù)Vue的構(gòu)造函數(shù),而將來我們在Vue-router中的一些實例方法中還要使用Vue的構(gòu)造函數(shù)
   * 把創(chuàng)建Vue實例時傳遞的Router對象注入到所有的vue實例上
   */
  static install (Vue) {
    // 判斷當前插件是否已經(jīng)被安裝
    if (VueRouter.install.installed) {
      return
    }
    VueRouter.install.installed = true

    // 把VUE的構(gòu)造函數(shù)記錄到全局變量中
    _Vue = Vue

    // 把創(chuàng)建Vue實例時傳遞的Router對象注入到所有的vue實例上
    // _Vue.prototype.$router = this.$options.router
    // 混入
    _Vue.mixin({
      beforeCreate() {
        if (this.$options.router) { // 實例的選項才有router,組件的選項沒有router
          _Vue.prototype.$router = this.$options.router
          this.$options.router.init() 
        }
      },
    })
  }

  constructor (options) {
    this.options = options // 記錄構(gòu)造函數(shù)中傳入的選項
    this.routeMap = {} // 把options中傳入的路由規(guī)則解析出來,存儲到routeMap, 鍵:儲存的是路由地址,值:存儲的是路由組件;
    this.data = _Vue.observable({
      current: window.location.pathname // 存儲當前路由地址
    })
  }

  init () {
    this.createRouteMap()
    this.initComponent(_Vue)
    this.initEvent()
  }

  // createRouteMap()的作用是把我們構(gòu)造函數(shù)中傳過來的選項中的路由規(guī)則,存儲到routeMap對象中
  createRouteMap () {
    // 遍歷所有的路由規(guī)則,把路由規(guī)則解析成鍵值對的形式,存儲到routeMap對象中
    this.options.routes.forEach(route => {
      this.routeMap[route.path] = route.component
    })
  }

  // initComponents
  initComponent (Vue) {
    Vue.component('router-link', {
      props: {
        to: String
      },
      render (h) {// h的作用是幫我們創(chuàng)建虛擬dom
        // 第一個參數(shù):選擇器,第二個參數(shù):屬性, 事件,第三個參數(shù):子元素
        return h('a', {
          attrs: {
            href: this.to
          },
          on: {
            click: this.clickHandler
          }
        }, [this.$slots.default])
      },
      methods: {
        clickHandler (e) {
          history.pushState({}, '', this.to)
          this.$router.data.current = this.to
          e.preventDefault()
        }
      }
      // template: '<a :href="to"><slot></slot></a>'
    })
    const self = this
    Vue.component('router-view', {
      render (h) {
        const component = self.routeMap[self.data.current]
        return h(component)
      }
    })
  }

  initEvent () {
    window.addEventListener('popstate', () => {
      this.data.current = window.location.pathname
    })
  }

  /**
   * 總結(jié):
   * 完整版本的vue包含運行時和編譯器,完整版本的vue里面的編譯器就是把我們的模版幫我們編輯成render函數(shù)
   * 運行時版本的vue在組件中我們要自己來寫render函數(shù),不支持template
   */
}

最后把router配置文件的vue-router改成自己定義的文件
import VueRouter from '../vuerouter/index'

注意

  • vue-cli創(chuàng)建的項目默認使用的是運行時版本的Vue.js,如果想切換成編譯器版本的Vue.js,需要修改vue-cli配置
    項目的根目錄創(chuàng)建vue.config.js文件,添加下面的代碼
module.exports = { runtimeCompiler: true }
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末肴茄,一起剝皮案震驚了整個濱河市侣监,隨后出現(xiàn)的幾起案子悲靴,更是在濱河造成了極大的恐慌,老刑警劉巖厢钧,帶你破解...
    沈念sama閱讀 217,657評論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件炸裆,死亡現(xiàn)場離奇詭異哄啄,居然都是意外死亡漠秋,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,889評論 3 394
  • 文/潘曉璐 我一進店門罕扎,熙熙樓的掌柜王于貴愁眉苦臉地迎上來聚唐,“玉大人丐重,你說我怎么就攤上這事「瞬椋” “怎么了扮惦?”我有些...
    開封第一講書人閱讀 164,057評論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長亲桦。 經(jīng)常有香客問我崖蜜,道長,這世上最難降的妖魔是什么客峭? 我笑而不...
    開封第一講書人閱讀 58,509評論 1 293
  • 正文 為了忘掉前任豫领,我火速辦了婚禮,結(jié)果婚禮上桃笙,老公的妹妹穿的比我還像新娘氏堤。我一直安慰自己沙绝,他們只是感情好搏明,可當我...
    茶點故事閱讀 67,562評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著闪檬,像睡著了一般星著。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上粗悯,一...
    開封第一講書人閱讀 51,443評論 1 302
  • 那天虚循,我揣著相機與錄音,去河邊找鬼样傍。 笑死横缔,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的衫哥。 我是一名探鬼主播茎刚,決...
    沈念sama閱讀 40,251評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼撤逢!你這毒婦竟也來了膛锭?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,129評論 0 276
  • 序言:老撾萬榮一對情侶失蹤蚊荣,失蹤者是張志新(化名)和其女友劉穎初狰,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體互例,經(jīng)...
    沈念sama閱讀 45,561評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡奢入,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,779評論 3 335
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了媳叨。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片腥光。...
    茶點故事閱讀 39,902評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡丁存,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出柴我,到底是詐尸還是另有隱情解寝,我是刑警寧澤,帶...
    沈念sama閱讀 35,621評論 5 345
  • 正文 年R本政府宣布艘儒,位于F島的核電站聋伦,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏界睁。R本人自食惡果不足惜觉增,卻給世界環(huán)境...
    茶點故事閱讀 41,220評論 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望翻斟。 院中可真熱鬧逾礁,春花似錦、人聲如沸访惜。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,838評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽债热。三九已至砾嫉,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間窒篱,已是汗流浹背焕刮。 一陣腳步聲響...
    開封第一講書人閱讀 32,971評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留墙杯,地道東北人配并。 一個月前我還...
    沈念sama閱讀 48,025評論 2 370
  • 正文 我出身青樓,卻偏偏與公主長得像高镐,于是被迫代替她去往敵國和親溉旋。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 44,843評論 2 354