Vue之路由(Router)

了解路由之前,我们需要先理解一个概念:单页应用。

单页应用

SPA(single page application):单一页面应用程序,只有一个完整的页面;它在加载页面时,不会加载整个页面,而是只更新某个指定的容器中内容。单页面应用(SPA)的核心之一是: 更新视图而不重新请求页面。

路由

这里的路由指的是SPA(单页应用)的路径管理器。vue的单页面应用将路径和组件映射起来,路由用于设定访问路径,由路径之间的切换,实现组件的切换。


路由模块的本质就是建立起 url 和页面之间的映射关系。

配置路由

在main.js中配置:

// 0. 导入 Vue 和 VueRouter,要调用 Vue.use(VueRouter)
import Vue from 'vue'
import VueRouter from 'vue-router'

Vue.use(VueRouter)

// 1. 定义 (路由) 组件。
// 可以从其他文件 import 进来
import Foo from "./views/Foo.vue";
import Bar from "./views/Bar.vue";

// 2. 定义路由
// 每个路由应该映射一个组件。 其中"component" 可以是
// 通过 Vue.extend() 创建的组件构造器,
// 或者,只是一个组件配置对象。
// 我们晚点再讨论嵌套路由。
const routes = [
  { path: '/foo', component: Foo },
  { path: '/bar', component: Bar }
]

// 3. 创建 router 实例,然后传 `routes` 配置
// 你还可以传别的配置参数, 不过先这么简单着吧。
const router = new VueRouter({
  routes // (缩写) 相当于 routes: routes
})

// 4. 创建和挂载根实例。
// 记得要通过 router 配置参数注入路由,
// 从而让整个应用都有路由功能
const app = new Vue({
  router
}).$mount('#app')

其中导入路由除了可以使用import再在 routes中定义,也可以直接定义

const routes = [
  { path: '/foo', component: () => import('./views/Foo.vue') },
  { path: '/bar', component: () => import('./views/Bar.vue') },
  { path: '/user', component: () => import('./views/User.vue') }
];

路由的使用

我们使用 <router-link> 标签来表示路由导航,使用 <router-view>表示路由出口

<template>
  <div id="app">
    <h1>Hello App!</h1>
    <p>
      <!-- 使用 router-link 组件来导航. -->
      <!-- 通过传入 `to` 属性指定链接. -->
      <!-- <router-link> 默认会被渲染成一个 `<a>` 标签 -->
      <router-link to="/foo">Go to Foo</router-link>
      <router-link to="/bar">Go to Bar</router-link>
    </p>
    <!-- 路由出口 -->
    <!-- 路由匹配到的组件将渲染在这里 -->
    <!-- App.vue 中的 <router-view></router-view> 是路由的最高级出口 -->
    <router-view></router-view>
  </div>
</template>

<router-link> 存在一个属性 to,这个属性指定的路径,根据我们配置的路由中路径对应的组件。

<router-view> 相对于一个插槽,它所在的位置将渲染路由匹配到的组件。

命名路由

路由是可以命名的,通过命名可以实现路由的指定。

// 0. 导入 Album、List、Add、Empty 三个组件

// 1. 定义路由
const routes = [
  { path: '/foo',
    name: 'fooName',
    component: () => import('./views/Foo.vue')
  }
];

通过命名跳转:

<!-- to 的值是一个对象而不是字符串,所以要在 to 前加 : -->
<router-link :to="{name: 'fooName'}">Go to Foo</router-link>

路由布局管理

在路由使用时,我们会用到多层路由

在使用路由时,同为路由内容的 banner没有因为新的路由导航而改变,原因是存在多次路由

分解:

App.vue组件(父组件)

<template>
  <div id="app">
    <h1>Hello App!</h1>
    <p>
      <router-link to="/album/list">Go to Album List</router-link>
      |
      <router-link to="/album/add">Go to Album Add</router-link>
    </p>
    <!-- 路由匹配到的组件将渲染在这里 -->
    <!-- 在本例中为 Album.vue 组件 -->
    <router-view></router-view>
  </div>
</template>

Album.vue组件(第一层子组件)

<template>
  <div id="album">
    <div class="banner">banner</div>
    <!-- 路由匹配到的组件将渲染在这里 -->
    <!-- 在本例中为 List.vue 组件或 Add.vue 组件 -->
    <router-view></router-view>
  </div>
</template>

List.vue,Add.vue组件(第二次子组件)略

const routes = [
  {
    path: '/album',
    component: Album,
    // children 属性可以用来配置下一级路由(子路由)
    children: [
      { path: 'list', component: List },
      { path: 'add', component: Add }
    ]
  }
];

总结来说,在路由中,可以在某一路由下添加 children 来表示次级路由,此时父路由要以 ‘/’ 开头以表示为根目录,在使用时,子路由和父路由间也要用 ‘/’ 隔开。

表示子路由时可以使用

path: 'list'
path: '/album/list'

这两种方式来表示,但如 /list 是不行的,加上‘/’会表示为根目录会导致子组件直接渲染。

注:存在空路由

{ path: '', component: Empty },

动态路由

我们会遇上多个地址对应一个路由的情况如用户页面 /user/100 和 /user/200 两个用户界面都使用了一个组件,此时我们需要动态路由。

import User from "./views/User.vue";

const routes = [
  // id 就是路径参数
  { path: '/user/:id', component: User }
]

id 为路径参数,一个“路径参数”前需要使用冒号 :标记。当 url 匹配到路由中的一个路径时,参数值会被设置到this.$route.params.id 里,可以在组件内读取到。

比如如果为 /user/200 则 this.$route.params.id 的值就是200

捕获404界面

当用户输入的url不属于任何一个路由时,我们一边使用404 NotFound组件来渲染 这里我们用通配符‘*’来匹配任意路径

import NotFound from "./views/NotFound.vue";

const routes = [
  {
    // 会匹配所有路径
    path: '*',
    component: NotFound
  }
]

注意:当使用通配符配置路由时,应用通配符的路由应该放在最后(路由匹配是按顺序的)

当使用了通配符时,可以使用

this.$route.params.pathMatch

来读取路径