Vue Router 是 Vue.js 官方提供的路由管理器,快速构建单页应用
本文将会使用 Vue CLI 创建脚手架,内容包括动态路径参数、嵌套路由、命名视图、懒加载。

引入 Vue Router

在使用 Vue CLI 创建脚手架时需要引入 Router

如果没有,需要安装 vue-router 依赖,然后修改 main.js 等,但很麻烦就懒得写了

脚手架中的相关文件

先介绍一下脚手架中与路由相关的文件

src/router/index.js
router 的配置文件,管理路由。这里展示一个简易版的

import Vue from "vue";
import VueRouter from "vue-router";
import Home from "../views/Home.vue";
import About from "../views/About.vue";

Vue.use(VueRouter);

const routes = [
  {
    path: "/",
    name: "home",
    component: Home
  },
  {
    path: "/about",
    name: "about",
    component: About
  }
];

const router = new VueRouter({
  mode: "history",
  base: process.env.BASE_URL,
  routes
});
export default router;

解释:
routes 规定了路由与组件之间的映射,页面可以 import 进来
router 是路由管理器,这里有路由模式、基地址的配置,并把 routes 传入进来

src/App.vue
它是单页应用的入口(我不知道怎么形容,总之就是所有组件都会渲染在这个页面中),贴一下关键部分

<template>
  <div id="app">
    <div id="nav">
      <router-link to="/">Home</router-link>
      <router-link to="/about">About</router-link>
    </div>
    <router-view />
  </div>
</template>

解释:
router-link 是类似超链接一样的东西
router-view 是组件将会被渲染到的地方

src/main.js
importsrc/router,并且在 Vue 对象实例化的时候传入,随便贴个大概代码

import Vue from "vue";
import App from "./App.vue";
import router from "./router";

Vue.config.productionTip = false;

new Vue({
  router,
  render: h => h(App)
}).$mount("#app");

动态路由匹配

很多时候,不同的 URL 可能会对应同一个页面,比如不同 ID 的用户页面都会使用同一个组件渲染
因此可以使用动态路径参数来解决这个问题

例如,router 的配置文件可以这样写

import Vue from "vue";
import VueRouter from "vue-router";
import Home from "../views/Home.vue";
import User from "../views/User.vue";

Vue.use(VueRouter);

const routes = [
  {
    path: "/",
    name: "home",
    component: Home
  },
  {
    path: "/user/:id",
    name: "user",
    component: User
  }
];

const router = new VueRouter({
  mode: "history",
  base: process.env.BASE_URL,
  routes
});

export default router;

然后 User.vue 长这样

<template>
  <div class="user">
    <h1>User ID: {{ $route.params.id }}</h1>
  </div>
</template>

这个东西使用 path-to-regexp 作为路径匹配引擎,支持正则匹配,这是它的 GitHub 链接
https://github.com/pillarjs/path-to-regexp

然后路由匹配是按照定义的顺序决定的,先定义的先匹配
因此可以在路由的最后配置 404 页面

嵌套路由

就是在子页面中使用路由

我们在路由配置中为 user 添加 profile 子路由,具体如下

import Vue from "vue";
import VueRouter from "vue-router";
import Home from "../views/Home.vue";
import User from "../views/User.vue";
import Profile from "../views/Profile.vue"

Vue.use(VueRouter);

const routes = [
  {
    path: "/",
    name: "home",
    component: Home
  },
  {
    path: "/user/:id",
    name: "user",
    component: User,
    children: [
      {
        path: "profile",
        component: Profile
      }
    ]
  }
];

const router = new VueRouter({
  mode: "history",
  base: process.env.BASE_URL,
  routes
});
export default router;

同时修改 User.vue,为它添加 router-view

<template>
  <div class="user">
    <h1>User ID: {{ $route.params.id }}</h1>
    <router-view />
  </div>
</template>

命名视图

有时候同一个页面中会有多个 router-view,我们希望在不同的视图中显示不同的东西

路由配置如下

import Vue from "vue";
import VueRouter from "vue-router";
import Home from "../views/Home.vue";
import User from "../views/User.vue";
import Profile from "../views/Profile.vue"
import Profile2 from "../views/Profile2.vue"

Vue.use(VueRouter);

const routes = [
  {
    path: "/",
    name: "home",
    component: Home
  },
  {
    path: "/user/:id",
    name: "user",
    component: User,
    children: [
      {
        path: "profile",
        components: {
          default: Profile,
          profile2: Profile2
        }
      }
    ]
  }
];

const router = new VueRouter({
  mode: "history",
  base: process.env.BASE_URL,
  routes
});
export default router;

User.vue 如下:

<template>
  <div class="user">
    <h1>User ID: {{ $route.params.id }}</h1>
    <router-view />
    <router-view name="profile2"/>
  </div>
</template>

路由懒加载

对于一个路由中的组件,可以在被访问的时候再加载
import 改成函数就行,比如下面这个:

import Vue from "vue";
import VueRouter from "vue-router";
import Home from "../views/Home.vue";
const User = () => import("../views/User.vue");
const Profile = () => import("../views/Profile.vue");
const Profile2 = () => import("../views/Profile2.vue");

Vue.use(VueRouter);

const routes = [
  {
    path: "/",
    name: "home",
    component: Home
  },
  {
    path: "/user/:id",
    name: "user",
    component: User,
    children: [
      {
        path: "profile",
        components: {
          default: Profile,
          profile2: Profile2
        }
      }
    ]
  }
];

const router = new VueRouter({
  mode: "history",
  base: process.env.BASE_URL,
  routes
});
export default router;

还可以把多个组件打包到一个 chunk 中,就能以 chunk 为单位异步加载,参考这个
https://webpack.js.org/guides/code-splitting/