详解vue-router 初始化时做了什么


Posted in Javascript onJune 11, 2018

最近因为业务需要,实现了一个简单的前端 router,正好也来看一下 vue router 是怎么实现的。这次先来一起看看 vue-router 初始化时做了什么。

vue router 的初始化使用步骤

我们首先来看 vue-router 的使用步骤,然后再分别去看各个步骤都发生了什么。

使用 vue-router 需要经过一下几个步骤:

引入 vue-router:

import VueRouter from 'vue-router';

利用 vue 的插件机制,加载 vue-router:

Vue.use(VueRouter);

实例化 VueRouter:

const router = new VueRouter({
routes
})

实例化 Vue:

const app = new Vue({
router
}).$mount('#app');

Vue 的插件机制

vue 提供了一个 use 方法,来加载插件:

Vue.use = function (plugin: Function | Object) {
 const installedPlugins = (this._installedPlugins || (this._installedPlugins = []));
 if (installedPlugins.indexOf(plugin) > -1) {
  return this;
 }

 // additional parameters
 const args = toArray(arguments, 1);
 args.unshift(this);
 if (typeof plugin.install === 'function') {
  plugin.install.apply(plugin, args);
 } else if (typeof plugin === 'function') {
  plugin.apply(null, args);
 }
 installedPlugins.push(plugin);
 return this;
}

该方法首先检查插件是否已经加载,如果已经加载,直接返回 this。

如果没有加载过,会取所有的参数,并将 this 放在第一个。优先执行 plugin.install 方法,若不能执行,则直接执行 plugin 自身。

最后将 plugin push 到插件列表中。

那么我们就需要看 VueRouter 的 install 方法做了什么,VueRouter 类定义在 src/index.js 文件中。

利用 vue 的插件机制,加载 vue-router

入口文件 index.js 对外 export 了一个 VueRouter 类。VueRouter 类包含了 router 的各种方法,我们直接先来看一下 install 方法。

install 方法在 index.js 中绑定在 VueRouter 类上:

import { install } from './install'
VueRouter.install = install

它的实际实现是在 ./install.js 中,install 方法主要做了以下几个事情:

1、设置了两个 mixin:beforeCreate 和 destroyed。

Vue.mixin({
 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)
 }
})

2、在 Vue 上绑定 $route 和 $router。

Object.defineProperty(Vue.prototype, '$router', {
 get () { return this._routerRoot._router }
})

Object.defineProperty(Vue.prototype, '$route', {
 get () { return this._routerRoot._route }
})

3、注册两个组件,View 和 Link。

Vue.component('RouterView', View)
Vue.component('RouterLink', Link)

4、设置 beforeRouteEnter、beforeRouteLeave 和 beforeRouteUpdate 的 merge 策略。merge 策略的介绍可以见 这里 ,简单来说就是有重复的值时如何合并。

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

实例化 VueRouter

我们来看一下 VueRouter 的构造函数。首先,constructor 会初始化一些属性:

this.app = null
this.apps = []
this.options = options
this.beforeHooks = []
this.resolveHooks = []
this.afterHooks = []
this.matcher = createMatcher(options.routes || [], this)

其中 matcher 比较重要,后面会详细说。

之后会决定使用哪种模式:

let mode = options.mode || 'hash'
this.fallback = mode === 'history' && !supportsPushState && options.fallback !== false
if (this.fallback) {
 mode = 'hash'
}
if (!inBrowser) {
 mode = 'abstract'
}
this.mode = mode

switch (mode) {
 case 'history':
  this.history = new HTML5History(this, options.base)
  break
 case 'hash':
  this.history = new HashHistory(this, options.base, this.fallback)
  break
 case 'abstract':
  this.history = new AbstractHistory(this, options.base)
  break
 default:
  if (process.env.NODE_ENV !== 'production') {
   assert(false, `invalid mode: ${mode}`)
  }
}

由于 history 模式中的pushstate方法还有一些浏览器没有支持。history 模式在浏览器不支持时会回退到hash模式。

之后根据不同模式选择实例化不同模式的history类,可以看到 hash 模式和 history 模式分别对应了 HashHistory 和 HTML5History 两个类。

此外,如果是服务器端渲染,需要进行 router 匹配来获取要渲染的页面。此时服务器环境中没有history api,因此要自行抽象实现一个,就是 AbstractHistory。

实例化 Vue

实例化为Vue 类时,会将 VueRouter 的实例传入,这个变量放在 this.$options.router 中。由于 vue router 时以插件形式引入的,因此 这个 this.$options.router 还是给 vue router 自身来用的。

vue router 初始化所做的事情就是这些,下篇博客我们来一起看一下 vue router 实际运行时发生了什么。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
JS中简单的实现像C#中using功能(有源码下载)
Jan 09 Javascript
jquery实现tr元素的上下移动示例代码
Dec 20 Javascript
JavaScript下的时间格式处理函数Date.prototype.format
Jan 27 Javascript
vue.js事件处理器是什么
Mar 20 Javascript
详解Vue Elememt-UI构建管理后台
Feb 27 Javascript
解决vue-cli项目打包出现空白页和路径错误的问题
Sep 04 Javascript
jQuery实现合并表格单元格中相同行操作示例
Jan 28 jQuery
你不可不知的Vue.js列表渲染详解
Oct 01 Javascript
element-ui 实现响应式导航栏的示例代码
May 08 Javascript
vue实现在线学生录入系统
May 30 Javascript
JavaScript多种图形实现代码实例
Jun 28 Javascript
jquery实现加载更多"转圈圈"效果(示例代码)
Nov 09 jQuery
node中间层实现文件上传功能
Jun 11 #Javascript
几个你不知道的技巧助你写出更优雅的vue.js代码
Jun 11 #Javascript
Vue.js 中取得后台原生HTML字符串 原样显示问题的解决方法
Jun 10 #Javascript
实例详解Node.js 函数
Jun 10 #Javascript
微信小程序实现倒计时调用相机自动拍照功能
Jun 10 #Javascript
深入浅析Vue中的Prop
Jun 10 #Javascript
vue项目部署上线遇到的问题及解决方法
Jun 10 #Javascript
You might like
兼容firefox,chrome的网页灰度效果
2011/08/08 PHP
php格式化json函数示例代码
2016/05/12 PHP
php版微信自动登录并获取昵称的方法
2016/09/23 PHP
PHP错误和异常处理功能模块示例
2016/11/12 PHP
无缝滚动改进版支持上下左右滚动(封装成函数)
2012/12/04 Javascript
jQuery中removeAttr()方法用法实例
2015/01/05 Javascript
javascript 动态创建表格的2种方法总结
2015/03/04 Javascript
使用requestAnimationFrame实现js动画性能好
2015/08/06 Javascript
JavaScript和jquery获取父级元素、子级元素、兄弟元素的方法
2016/06/05 Javascript
HTML5canvas 绘制一个圆环形的进度表示实例
2016/12/16 Javascript
详解网站中图片日常使用以及优化手法
2017/01/09 Javascript
JavaScript 详解预编译原理
2017/01/22 Javascript
详解node HTTP请求客户端 - Request
2017/05/05 Javascript
vue better scroll 无法滚动的解决方法
2018/06/07 Javascript
vscode中vue-cli项目es-lint的配置方法
2018/07/30 Javascript
cdn模式下vue的基本用法详解
2018/10/07 Javascript
[01:00:14]2018DOTA2亚洲邀请赛 4.6 淘汰赛 VP vs TNC 第三场
2018/04/10 DOTA
python cookielib 登录人人网的实现代码
2012/12/19 Python
Python处理RSS、ATOM模块FEEDPARSER介绍
2015/02/18 Python
Python多线程编程(五):死锁的形成
2015/04/05 Python
Python中的random()方法的使用介绍
2015/05/15 Python
快速入手Python字符编码
2016/08/03 Python
python实现将读入的多维list转为一维list的方法
2018/06/28 Python
python读取txt文件,去掉空格计算每行长度的方法
2018/12/20 Python
python Django编写接口并用Jmeter测试的方法
2019/07/31 Python
Python函数__new__及__init__作用及区别解析
2020/08/31 Python
英国复古皮包品牌:Beara Beara
2018/07/18 全球购物
美国狗旅行和户外用品领先供应商:kurgo
2020/08/18 全球购物
校园之星获奖感言
2014/01/29 职场文书
人力资源管理专业自荐书范文
2014/02/10 职场文书
平安工地汇报材料
2014/08/19 职场文书
2014年保卫科工作总结
2014/12/05 职场文书
整改通知书
2015/04/20 职场文书
2015年纪委工作总结
2015/05/13 职场文书
VUE递归树形实现多级列表
2022/07/15 Vue.js
Java使用HttpClient实现文件下载
2022/08/14 Java/Android