Vue学习笔记——Vue-Router的使用

3,961 阅读6分钟

Vue-Router

  1. 理解: 一个路由(route)就是一组映射关系(key - value),多个路由需要路由器(router)进行管理。
  2. 前端路由:key是路径,value是组件。

基本使用

  1. 安装vue-router,命令:npm i vue-router

  2. 应用插件:Vue.use(VueRouter)

  3. 编写router配置项:

    //引入VueRouter
    import VueRouter from 'vue-router'
    //引入Luyou 组件
    import About from '../components/About'
    import Home from '../components/Home'//创建router实例对象,去管理一组一组的路由规则
    const router = new VueRouter({
        routes:[
            {
                path:'/about',
                component:About
            },
            {
                path:'/home',
                component:Home
            }
        ]
    })
    ​
    //暴露router
    export default router
    
  4. 实现切换(active-class可配置高亮样式)

    <router-link active-class="active" to="/about">About</router-link>
    
  5. 指定展示位置

    <router-view></router-view>
    

注意:

  1. 路由组件通常存放在pages文件夹,一般组件通常存放在components文件夹。
  2. 通过切换,“隐藏”了的路由组件,默认是被销毁掉的,需要的时候再去挂载。
  3. 每个组件都有自己的$route属性,里面存储着自己的路由信息。
  4. 整个应用只有一个router,可以通过组件的$router属性获取到。

嵌套路由(多级路由)

  1. 配置路由规则,使用children配置项:

    routes:[
        {
            path:'/about',
            component:About,
        },
        {
            path:'/home',
            component:Home,
            children:[ //通过children配置子级路由
                {
                    path:'news', //此处一定不要写:/news
                    component:News
                },
                {
                    path:'message',//此处一定不要写:/message
                    component:Message
                }
            ]
        }
    ]
    
  2. 跳转(要写完整路径):

    <router-link to="/home/news">News</router-link>
    

路由的query参数

  1. 传递参数

    <!-- 跳转并携带query参数,to的字符串写法 -->
    <router-link :to="/home/message/detail?id=666&title=你好">跳转</router-link>
                    
    <!-- 跳转并携带query参数,to的对象写法 -->
    <router-link 
        :to="{
            path:'/home/message/detail',
            query:{
               id:666,
                title:'你好'
            }
        }"
    >跳转</router-link>
    
  2. 接收参数:

    $route.query.id
    $route.query.title
    

命名路由(name属性)

  1. 作用:可以简化路由的跳转。

  2. 如何使用

    1. 给路由命名:

      {
          path:'/demo',
          component:Demo,
          children:[
              {
                  path:'test',
                  component:Test,
                  children:[
                      {
                            name:'hello' //给路由命名
                          path:'welcome',
                          component:Hello,
                      }
                  ]
              }
          ]
      }
      
    2. 简化跳转:

      <!--简化前,需要写完整的路径 -->
      <router-link to="/demo/test/welcome">跳转</router-link><!--简化后,直接通过名字跳转 -->
      <router-link :to="{name:'hello'}">跳转</router-link><!--简化写法配合传递参数 -->
      <router-link 
          :to="{
              name:'hello',
              query:{
                 id:666,
                  title:'你好'
              }
          }"
      >跳转</router-link>
      

路由的params参数

  1. 配置路由,声明接收params参数

    {
        path:'/home',
        component:Home,
        children:[
            {
                path:'news',
                component:News
            },
            {
                component:Message,
                children:[
                    {
                        name:'xiangqing',
                        path:'detail/:id/:title', //使用占位符声明接收params参数
                        component:Detail
                    }
                ]
            }
        ]
    }
    
  2. 传递参数

    <!-- 跳转并携带params参数,to的字符串写法 -->
    <router-link :to="/home/message/detail/666/你好">跳转</router-link>
                    
    <!-- 跳转并携带params参数,to的对象写法 -->
    <router-link 
        :to="{
            name:'xiangqing',
            params:{
               id:666,
                title:'你好'
            }
        }"
    >跳转</router-link>
    

    特别注意:路由携带params参数时,若使用to的对象写法,则不能使用path配置项,必须使用name配置!

  3. 接收参数:

    $route.params.id
    $route.params.title
    

路由的props配置

作用:让路由组件更方便的收到参数

{
    name:'xiangqing',
    path:'detail/:id',
    component:Detail,
​
    //第一种写法:props值为对象,该对象中所有的key-value的组合最终都会通过props传给Detail组件
    // props:{a:900}//第二种写法:props值为布尔值,布尔值为true,则把路由收到的所有params参数通过props传给Detail组件
    // props:true
    
    //第三种写法:props值为函数,该函数返回的对象中每一组key-value都会通过props传给Detail组件
    props(route){
        return {
            id:route.query.id,
            title:route.query.title
        }
    }
}

<router-link>的replace属性

  1. 作用:控制路由跳转时操作浏览器历史记录的模式
  2. 浏览器的历史记录有两种写入方式:分别为pushreplacepush是追加历史记录,replace是替换当前记录。路由跳转时候默认为push
  3. 如何开启replace模式方式一:<router-link replace .......>News</router-link>
  4. 如何开启replace模式方式二:<router-link :replace="true" .......>News</router-link>

router-link的属性

组件的属性有:

to 、replace、 append、 tag、 active-class、 exact 、 event、 exact-active-class

1、to(必选参数):类型string/location

表示目标路由的链接,该值可以是一个字符串,也可以是动态绑定的描述目标位置的对象

2、tag

类型: string

默认值: "a"

如果想要 渲染成某种标签,例如

  • 。 于是我们使用 tag prop 类指定何种标签,同样它还是会监听点击,触发导航。

    3、active-class

    类型: string

    默认值: "router-link-active"

    设置 链接激活时使用的 CSS 类名。默认值可以通过路由的构造选项 linkActiveClass 来全局配置。

    4、exact-active-class

    类型: string

    默认值: "router-link-exact-active"

    配置当链接被精确匹配的时候应该激活的 class。注意默认值也是可以通过路由构造函数选项 linkExactActiveClass 进行全局配置的。

    5、exact

    类型: boolean

    默认值: false

    "是否激活" 默认类名的依据是 inclusive match (全包含匹配)。 举个例子,如果当前的路径是 /a 开头的,那么 也会被设置 CSS 类名。

    6、event 类型: string | Array

    默认值: 'click'

    声明可以用来触发导航的事件。可以是一个字符串。

    7、replace

    类型: boolean

    默认值: false

    设置 replace 属性的话,当点击时,会调用 router.replace() 而不是 router.push(),于是导航后不会留下 history 记录。

    8、append

    类型: boolean

    默认值: false

    设置 append 属性后,则在当前 (相对) 路径前添加基路径

    编程式路由导航

    router.push(location, onComplete?, onAbort?)

    当你点击 <router-link> 时,这个方法会在内部调用,所以说,点击 <router-link :to="..."> 等同于调用 router.push(...)

    该方法的第一个参数可以是一个字符串路径,或者一个描述地址的对象

    // 字符串
    router.push('home')
    ​
    // 对象
    router.push({ path: 'home' })
    ​
    // 命名的路由
    router.push({ name: 'user', params: { userId: '123' }})
    ​
    // 带查询参数,变成 /register?plan=private
    router.push({ path: 'register', query: { plan: 'private' }})
    

    注意:如果提供了 pathparams 会被忽略,上述例子中的 query 并不属于这种情况。取而代之的是下面例子的做法,你需要提供路由的 name 或手写完整的带有参数的 path

    const userId = '123'
    router.push({ name: 'user', params: { userId }}) // -> /user/123
    router.push({ path: `/user/${userId}` }) // -> /user/123
    // 这里的 params 不生效
    router.push({ path: '/user', params: { userId }}) // -> /user
    

    同样的规则也适用于 router-link 组件的 to 属性。

    可选的在 router.pushrouter.replace 中提供 onCompleteonAbort 回调作为第二个和第三个参数。这些回调将会在导航成功完成 (在所有的异步钩子被解析之后) 或终止 (导航到相同的路由、或在当前导航完成之前导航到另一个不同的路由) 的时候进行相应的调用。

    可以省略第二个和第三个参数,此时如果支持 Promise,router.pushrouter.replace 将返回一个 Promise。

    router.replace(location, onComplete?, onAbort?)

    router.push 很像,唯一的不同就是,它不会向 history 添加新记录,而是跟它的方法名一样 —— 替换掉当前的 history 记录。

    router.go(n)

    这个方法的参数是一个整数,意思是在 history 记录中向前或者后退多少步,类似 window.history.go(n)

    // 在浏览器记录中前进一步,等同于 history.forward()
    router.go(1)
    ​
    // 后退一步记录,等同于 history.back()
    router.go(-1)
    ​
    // 前进 3 步记录
    router.go(3)
    ​
    // 如果 history 记录不够用,那就默默地失败呗
    router.go(-100)
    router.go(100)
    

    router.push(location, onComplete?, onAbort?)
    router.push(location).then(onComplete).catch(onAbort)
    router.replace(location, onComplete?, onAbort?)
    router.replace(location).then(onComplete).catch(onAbort)
    router.go(n)
    router.back()
    router.forward()
    

    函数式导航总结

    1. 作用:不借助<router-link>实现路由跳转,让路由跳转更加灵活

    2. 具体编码:

      //$router的两个API
      this.$router.push({
          name:'xiangqing',
              params:{
                  id:xxx,
                  title:xxx
              }
      })
      ​
      this.$router.replace({
          name:'xiangqing',
              params:{
                  id:xxx,
                  title:xxx
              }
      })
      this.$router.forward() //前进
      this.$router.back() //后退
      this.$router.go() //可前进也可后退
      

    缓存路由组件

    1. 作用:让不展示的路由组件保持挂载,不被销毁。

    2. 具体编码:include 内写需要缓存的组件名,若不写则当前路由视图内的所有组件均被缓存

      <!-- 缓存多个路由组件 -->
      <!-- <keep-alive :include="['News','Message']"> -->
                      
      <!-- 缓存一个路由组件 -->
      <keep-alive include="News"> 
          <router-view></router-view>
      </keep-alive>
      

    两个新的生命周期钩子

    1. 作用:路由组件所独有的两个钩子,用于捕获路由组件的激活状态。

    2. 具体名字:

      1. activated路由组件被激活时触发。
      2. deactivated路由组件失活时触发。

    路由守卫

    1. 作用:对路由进行权限控制

    2. 分类:全局守卫、独享守卫、组件内守卫

    3. 全局守卫:

      //全局前置守卫:初始化时执行、每次路由切换前执行
      router.beforeEach((to,from,next)=>{
          console.log('beforeEach',to,from)
          if(to.meta.isAuth){ //判断当前路由是否需要进行权限控制
              if(localStorage.getItem('school') === 'atguigu'){ //权限控制的具体规则
                  next() //放行
              }else{
                  alert('暂无权限查看')
                  // next({name:'guanyu'})
              }
          }else{
              next() //放行
          }
      })
      ​
      //全局后置守卫:初始化时执行、每次路由切换后执行
      router.afterEach((to,from)=>{
          console.log('afterEach',to,from)
          if(to.meta.title){ 
              document.title = to.meta.title //修改网页的title
          }else{
              document.title = 'vue_test'
          }
      })
      
    4. 独享守卫:

      beforeEnter(to,from,next){
          console.log('beforeEnter',to,from)
          if(to.meta.isAuth){ //判断当前路由是否需要进行权限控制
              if(localStorage.getItem('school') === 'atguigu'){
                  next()
              }else{
                  alert('暂无权限查看')
                  // next({name:'guanyu'})
              }
          }else{
              next()
          }
      }
      
    5. 组件内守卫:

      //进入守卫:通过路由规则,进入该组件时被调用
      beforeRouteEnter (to, from, next) {
      },
      //离开守卫:通过路由规则,离开该组件时被调用
      beforeRouteLeave (to, from, next) {
      }
      

    路由器的两种工作模式

    1. 对于一个url来说,什么是hash值?—— #及其后面的内容就是hash值。

    2. hash值不会包含在 HTTP 请求中,即:hash值不会带给服务器。

    3. hash模式:

      1. 地址中永远带着#号,不美观 。
      2. 若以后将地址通过第三方手机app分享,若app校验严格,则地址会被标记为不合法。
      3. 兼容性较好。
    4. history模式:

      1. 地址干净,美观 。
      2. 兼容性和hash模式相比略差。
      3. 应用部署上线时需要后端人员支持,解决刷新页面服务端404的问题。