vue-router 还给路由排了序?解析路由匹配,vue-router Matcher 解析(三)

简介: 之前的两篇文章讲了, vue-router 的 Matcher 对初始的 routes 进行了标准化(normalized)处理以及别名(alias)处理,但还有matcher 处理 route 的匹配

前情提要

之前的两篇文章讲了, vue-router 的 Matcher 对初始的 routes 进行了标准化(normalized)处理以及别名(alias)处理,详情链接参考文章尾部

本篇文章会介绍 matcher 处理 route 的匹配部分, 即 vue-router 文档的路由的匹配语法

注:本篇文章关于 matcher 的解析和源码均对应 vue-router4 即 vue3 版本的 router

注意区分文章中的, routes 和 route, route 指 { path: "/", Redirect: "/test" }, routes 指 [route,route,...]

matcher 数组

之前提到过, 类似 routes 这样的数组其实是一种多叉树结构, 然后整个 matcher 数组生成的过程中使用了递归式的多叉树遍历,并在遍历过程中处理了别名以及别名子路由,将子路由的 path 修改为正确的, 比如 /a/1, 其实这个在 matcher 里面都是一个次要的功能,真正的功能是去生成 matchers 数组,对,虽然之前文章花了重点介绍 addRoute,但是不要被它 语义 给迷惑了,它的真正目的是给 matchers 添加 route 转化的 matcher(小细节)

又是一个小细节

// routes 其实是多叉树
const routes = [
  {
    path: "/",
    component: Home,
  },
  {
    path: "/a",
    alias: "/ab",
    component: null,
    children: [{ component: null, path: "1" }],
  },
  {
    path: "/b",
    component: Home,
  },
];

区分路由

前面说了 addRoute 目的是给 matchers 添加 route 转化的 matcher, 这个证明在于 addRoute 的底部 router/src/matcher/index.ts 179

function addRoute(...) {
  // ...
  insertMatcher(matcher);
}
// ...
function insertMatcher(matcher: RouteRecordMatcher) {}

insertMatcher() 就是今天这篇文章的主角

为什么说区分路由? 我们先看一下 insertMatcher 的源码 router/src/matcher/index.ts 214

function insertMatcher(matcher: RouteRecordMatcher) {
  let i = 0;
  while (
    i < matchers.length &&
    comparePathParserScore(matcher, matchers[i]) >= 0
  )
    i++;
  // 关键的下面三行
  matchers.splice(i, 0, matcher);
  if (matcher.record.name && !isAliasRecord(matcher))
    matcherMap.set(matcher.record.name, matcher);
}

insertMatcher() 主要作用就是找到 matcher(由 route 转化的) 在 matchers 中的位置, 这也就是我标题为什么说 matchers 是一个有顺序的数组, 而区分路由就是指 matchers 和 matcherMap 两个数据结构, 一个数组一个哈希表, 在上面标注的源码中有一个性能优化的点, 就是命名路由的 matcher 是有另一份存放在 matcherMap 里面的, 这很重要, 因为 Map 这种数据结构的查询时间复杂度是 O(1), 是很快的, 所以我们在编写路由的, 应该尽可能的使用命名路由, 它的查询时间远远快于普通路由, 普通路由的查询时间复杂度是 O(n), matcher 里面的 resolve 就是我们匹配路由是调用的方法, 里面其实就写了命名路由和普通路由的匹配方式

// router/src/matcher/index.ts 283 resolve()
matcher = currentLocation.name
  ? matcherMap.get(currentLocation.name)
  : matchers.find((m) => m.re.test(currentLocation.path));

为了方便理解, 请参考下面原始 routes 的数据结构和 matchers 的数据结构

const routes = [
  {
    path: "/",
    component: Home,
  },
  {
    path: "/a",
    alias: "/ab",
    component: null,
    children: [{ component: null, path: "1" }],
  },
  {
    path: "/b",
    component: Home,
  },
];

matchers

1.png

前排提示, score 是一个伏笔

路径排名

前面说了, insertMatcher() 是通过 while() 遍历找到 matchers 的正确的插入位置的, 源码如下

// 生成 matchers 的代价是 O(n^2)
while (i < matchers.length && comparePathParserScore(matcher, matchers[i]) >= 0)
  i++;
// 模拟 matchers 的生成过程
1. [/]
2. [/,/a/1]
3. [/,/a,/a/1]
4.
...

小细节来了, 这个 matchers 它其实是一个有顺序的数组, 不是乱排的, 它有个路径排名的功能, 里面核心有两个函数 comparePathParserScorecompareScoreArray

简单介绍一下 matcher 它这个排名的实现, 生成 matcher 的时候, 根据你传入的 route 的 path 打分, 然后根据这个分的高低去给你排序看你适合排在哪里, 它的主要目的是什么呢?这就要说到这个设计者在北京分享的一个讲座的小细节了

const routes = [
  {
    path: "/movie/:id",
  },
  {
    path: "/movie/new",
  },
];

像上面这个路由, 如果你传入 /movie/new 其实两个都是满足条件的, 但是很明显第二个才是我们想要的, 所以为了正确的匹配我们必须让 /movie/new 在之前 /movie/:id 被匹配到, 注意, 我们前面说过, 普通路由匹配的方式就是 matchers.find(), 就是从 0 到 n 匹配是返回第一个适合的, 所以为什么需要路径排名? 为什么 matchers 是一个有顺序的数组? 就是要让 /movie/new 这种静态路由能够比 /movie/:id 这种动态路由先匹配到, 先静后动

回到它的底层实现, 看之前关于 matchers 的数据结构截图, 里面有个属性叫 score, 它里面存放的就是根据 path 打分得到的 number 数组, 像 /movie/new 这种静态路由的分数要比 /movie/:id 这种动态路由高, 因此匹配的时候就不会出错, 可以参考下面 Chrome 里面的调试截图

2.png

3.png

4.png

剩余的排名就是按照源码(指你写的 routes)的顺序, 比如

// routes
const routes = [
  {
    path: "/a",
  },
  {
    path: "/b",
  },
];
// matchers
const matchers = ["/a", "/b"];

说实话,我是有点失望的, vue-router matcher 的 diff 算法, 和存放 matcher 的数组似乎不是我认识的数据结构和算法,我一开始看到 vue-router 的几个特征, 1. N 叉树, 2. matcher 用数组存放 3. matchers 生成顺序依靠 vue-router 自定义得分, 几乎那两个数据结构是呼之欲出,但是竟然不是,这两个数据结构就是, 二叉搜索树(BST)二叉堆(优先级队列), 为什么认为是这两个数据结构,一是我认为二叉搜索树的查找效率为 O(logn) 很快, 二就是我单纯的看到数组以为会用到二叉堆

但是回过头一想, routes 是 N 叉树, 可能上面这两种数据结构并不合适, 所以纯属是我刷题刷魔怔了

image.png

总结

vue-router matcher 基本都讲完了, 算是非常细节了, 百度 google 都找不到几篇讲 matcher 的, 更不要说是 vue-router4 的源码解析, 不知道还会不会继续写关于 matcher 部分的内容, 其实 matcher 部分还有 path 的解析没有细细展开讲解, 可能还会再出一篇, 剩下的就是 matcher 和 history 以及 component 的联调解析了

vue-router matcher 部分的 path ranking(路径排名) 就先讲到这里,如果你对 vue-router4 的源码感兴趣的话,觉得文章讲的还行,可以看之后发的几篇源码解析(关注就能看到后续更新, 欢迎点赞催更), Matcher 部分文章如下

  1. vue-router 如何找到待渲染的 vue 组件?vue-router Matcher 解析(一)) - 标准化 route
  2. vue-router 如何找到待渲染的 vue 组件?详解 route 别名处理,vue-router Matcher 解析(二 - 别名处理
  3. [vue-router 如何匹配路由?解析路径排名,vue-router Matcher 解析(三)] - 路径排名
相关文章
|
19天前
|
网络协议 物联网 网络安全
|
19天前
|
JavaScript 前端开发 网络架构
Vue3项目中使用vue-router
Vue3项目中使用vue-router
58 0
|
9天前
|
域名解析 负载均衡 网络协议
【域名解析 DNS 专栏】如何利用 DNS 实现地理位置路由
【5月更文挑战第25天】利用DNS进行地理位置路由,能根据用户位置导向最近服务器,提升网络效率和用户体验。通过判断请求IP地址归属地,DNS返回相应区域的服务器IP。简单示例代码展示了如何实现此功能,但实际应用需借助专业数据库和处理网络复杂性的工具。尽管面临挑战,如代理服务器和网络环境影响,但DNS地理位置路由仍是优化网络性能的关键技术,对于内容分发和负载均衡具有重要意义。随着技术进步,这一领域的潜力将持续挖掘。
|
17天前
Vue3+Vite+Pinia+Naive后台管理系统搭建之三:vue-router 的安装和使用
Vue3+Vite+Pinia+Naive后台管理系统搭建之三:vue-router 的安装和使用
25 0
|
17天前
|
网络架构
Vue3 系列:vue-router
Vue3 系列:vue-router
21 2
|
19天前
|
监控 网络协议 安全
【亮剑】当设备IP能ping通但无法上网时,可能是DNS解析、网关/路由设置、防火墙限制、网络配置错误或ISP问题
【4月更文挑战第30天】当设备IP能ping通但无法上网时,可能是DNS解析、网关/路由设置、防火墙限制、网络配置错误或ISP问题。解决步骤包括检查网络配置、DNS设置、网关路由、防火墙规则,以及联系ISP。预防措施包括定期备份配置、更新固件、监控网络性能和实施网络安全策略。通过排查和维护,可确保网络稳定和安全。
|
19天前
|
资源调度 JavaScript 前端开发
【vue】vue中的路由vue-router,vue-cli脚手架详细使用教程
【vue】vue中的路由vue-router,vue-cli脚手架详细使用教程
|
19天前
|
JavaScript 前端开发 UED
AngularJS路由管理:深度解析$routeProvider的应用与实践
【4月更文挑战第28天】本文深入解析AngularJS的$routeProvider,它是AngularJS路由系统的关键,用于定义应用的视图和路径。通过routeProvider,开发者能根据URL变化动态加载内容,实现单页应用效果。配置$routeProvider涉及导入angular-route.js,注入&quot;ngRoute&quot;依赖,并使用when方法定义路由规则。ng-view指令用于显示路由打开的页面,而otherwise方法处理未定义路由,提供默认响应。$routeProvider使导航体验优化,助力构建高效Web应用。
|
19天前
|
网络协议 SDN 数据安全/隐私保护
|
19天前
vue3使用vue-router嵌套路由(多级路由)
vue3使用vue-router嵌套路由(多级路由)
129 0

推荐镜像

更多