vue-router浅析

在进行路由创建时,我们需要一个router对所有的路由进行管理工作以及确认采取哪种路由模式。 route为路由对象,它包括path、component、name等

new Router

new router时进行matcher的创建和路由模式的确定
history采用浏览器历史调用栈的方式、 hash模式则使用带#URL格式

var VueRouter = function VueRouter (options) {
  if ( options === void 0 ) options = {};

  this.app = null;
  this.apps = [];
  this.options = options;
  this.beforeHooks = [];
  this.resolveHooks = [];
  this.afterHooks = [];
  // matcher对象包含用于路由匹配的方法、还有addRoutes添加路由方法
  // 调用该方法里会 进行createRouteMap方法的调用, 生成routeMap
  this.matcher = createMatcher(options.routes || [], this);

  var 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));
      }
  }
};

createRouteMap

以下面一个简单的示例为例

const Foo = { template: '<div>foo</div>' }
const Bar = { template: '<div>bar</div>' }
const Sub = { template: '<div>Sub</div>' }
...

const routes = [
  { path: '/foo', component: Foo , children: [
    {
      path: '/sub',
      component: Sub
    }
  ]},
  { path: '/bar', component: Bar }
]

经过createRouteMap后返回的结果如下


router-ceateMap.png

createRouteMap将路由对象进行了拍平处理, 生成pathMap映射。
path属性对应当前的路由路径
components当前路由对应的模板
regex进行路径的匹配工作,同时对动态路由进行params的获取

全局挂载

我们知道在使用$router时可获取到router实例对象, 其实现的代码如下
通过beforeCreate在Vue组件上混入_routerRoot根路由, 方法$router时实际上是通过this._routerRoot._router获取到的。

Vue.mixin({
    beforeCreate: function 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: function destroyed () {
      registerInstance(this);
    }
  });

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

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

思考一下在进行路由切换的时候经历了一些什么, 略去细节的话大体如下
1)比如我们会先点击一个link,
2)link组件会触发handler方法
3)进行当前路由属性的变更,
4)路由的变更最终引起view视图更新工作。
更具体一些的实现如下

RouterLink组件

绑定事件方法如下

var handler = function (e) {
      if (guardEvent(e)) {
        if (this$1.replace) {
          router.replace(location, noop);
        } else {
          router.push(location, noop);
        }
      }
    };

以history模式为例
其中router.push的调用轨迹 router.push -> history.push -> transitionTo -> confirmTransition
-> runQueue -> onComplete -> updateRoute -> this.cb(route) -> app._route = route;

  1. router.push中进行了history.push的调用工作。
  2. history.push获取当前的路由对象作为fromRoute,可在导航守卫中进行使用
  3. transitionTo为路由切换的入口函数, 不管是link点击、history前进后退或者直接调用push方法最终都是以该方法作为导航切换的入口。
    在该函数中会进行route对象的匹配获取, 然后进入到confirmTransition方法
  4. confirmTransition创建调用队列包括 extractLeaveGuards 、beforeHooks、extractUpdateHooks、activated.、resolveAsyncComponents。
    创建一个迭代方法用于队列的调用,队列事件完成后进行回调函数的处理
  5. onComplete包括 updateRoute、 pushState、afterHooks等方法
  6. updateRoute会调用history.listen中添加的cb方法从而将app上挂载的_route进行更新,触发set监听和获取当前路由。
    pushState最终调用 history.pushState方法向历史堆栈中添加一条状态

RouterView组件

在render的时候从var route = parent.$route;获取当前的route对象, 该组件是函数式组件会从parent中获取route。
_route属性为响应式的, 当它发生变化会触发view视图的更新。

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

_route响应式实现的方法

Vue.util.defineReactive(this, '_route', this._router.history.current);
 var route = parent.$route;
 ...
 history.listen(function (route) {
    this$1.apps.forEach(function (app) {
      app._route = route;
    });
  });

通过defineReactive, 将_route设置为响应式的。 get进行依赖收集, set时触发view更新。

路由事件绑定及变化过程

var eventType = supportsPushState ? 'popstate' : 'hashchange';
    window.addEventListener(
      eventType,
      handleRoutingEvent
    );
    this.listeners.push(function () {
      window.removeEventListener(eventType, handleRoutingEvent);
    });

transitionTo -> (router.match),
updateRoute用于更新当前路由 -> app._route = route;
confirmTransition ->resolveQueue
当活动历史记录条目更改时,将触发popstate事件。需要注意的是调用history.pushState()或history.replaceState()不会触发popstate事件。只有在做出浏览器动作时,才会触发该事件,如用户点击浏览器的回退按钮
在使用history模式时需考虑404问题。

url的hash发生变更时会触发 hashchange方法

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,794评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,050评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,587评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,861评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,901评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,898评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,832评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,617评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,077评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,349评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,483评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,199评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,824评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,442评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,632评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,474评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,393评论 2 352

推荐阅读更多精彩内容