目录​​​​​​​

一、vue router的安装与简单使用

1、Vue Router的安装

2、一个简单的vue router使用示例

二、带参数的动态路由

1、路由参数匹配

2、路由匹配的语法规则

3、路由的嵌套

三、页面导航

1、使用路由的方法

四、关于路由的命名

1、使用名称进行路由切换

2、路由视图命名

3、使用别名

4、路由重定向

五、关于路由传参

六、路由导航守卫

1、定义全局导航守卫

2、为特定的路由注册注册导航守卫

七、动态路由

1、动态添加与删除路由


一、vue router的安装与简单使用

1、Vue Router的安装

使用vue cli创建一个示例的vue项目工程,使用终端在根目录下执行如下指令来安装vue-router

npm install vue-router@4

2、一个简单的vue router使用示例

在工程的components文件夹下新建两个文件,分别命名为demo1.vue和demo2.vue

demo1.vue

<template><h1>我是页面一</h1>
</template>
<script>export default {}
</script>

demo2.vue

<template><h1>我是页面二</h1>
</template>
<script>export default {}
</script>

main.js

import { createApp } from 'vue'
import App from './App.vue'
import './index.css'
//导入vue-router模块中的createRouter和createWebHashHistory方法
import { createRouter, createWebHashHistory } from 'vue-router'
//导入路由需要用到的自定义组件
import Demo1 from './components/Demo1.vue'
import Demo2 from './components/Demo2.vue'
const app = createApp(App)
//定义路由
const routes = [{ path: '/demo1', component: Demo1 },{ path: '/demo2', component: Demo2 }
]
//创建路由对象
const router = createRouter({history: createWebHashHistory(),routes: routes
})
//注册路由
app.use(router)
app.mount('#app')

运行上面代码

二、带参数的动态路由

1、路由参数匹配

在工程的components文件夹下新建一个名为User.vue的文件

<template><h2>姓名:{{$route.params.username}}</h2><h2>id:{{$route.params.id}}</h2>
</template>

在组件内部可以使用$route属性获取全局的路由对象,路由中定义的参数可以在此对象的params属性中获取到

import User from './components/User.vue'

{ path: '/user/:username/:id', component: User }

在定义路由的路径path时,使用冒号来标记参数,username和id都是路由的参数

现在在浏览器中输入如下格式的地址

http://localhost:3000/#/user/小王/8888

结果:

需要注意的是,在使用带参数的路由时,对应相同组件的路由在进行导航切换时,相同的组件并不会被销毁再创建,这种复用机制使得页面的加载效率更高。但这也表明,页面切换时,组件的生命周期方法都不会被再次调用,如果需要通过路由参数来请求数据,之后渲染页面需要特别注意,不能再生命周期方法中进行数据请求逻辑。

App.vue组件的模板修改如下

<router-link to="/user/小王/8888">小王</router-link>
<br>
<router-link to="/user/小李/6666">小李</router-link>

user.vue的代码修改如下

mounted() {
alert(`组件加载,请求数据。路由参数为name${this.$route.params.username},路由参数为id${this.$route.params.id}`)
}
}

alert组件只有在第一次加载时才会调用,在同一组件中切换路由时, mounted不会被调用

对于这种场景,我们可以采用导航守卫的方法来处理,每次路由参数都会更新,都会回调守卫函数。

user.vue的代码修改如下

//to是更新后的路由对象,from是更新前的路由对象beforeRouteUpdate(to, from) {alert(`组件加载,请求数据。路由参数为name${to.params.username},路由参数为id${to.params.id}`)}

2、路由匹配的语法规则

在进行路由参数匹配时,vue router允许参数内部使用正则表达式来匹配。

user组件的代码修改

<h1>用户中心</h1>

<h2>姓名:{{$route.params.username}}</h2>

新建名为UserSetting.vue的文件

<h1>用户设置</h1>

<h2>id:{{$route.params.id}}</h2>

在main.js文件中定义路由

{ path: '/user/:username', component: User },

{ path: '/user/:id', component: UserSetting }

上面的代码中定义的两个路由参数除了参数名不同外,格式完全一样。这种情况下,我们无法访问用户设置页面,所有符合UserSetting组件的路由规则同时也符合User组件。为了解决这一问题,最简单的方式是加一个静态的前缀路径。

{ path: '/user/info/:username', component: User },

{ path: '/user/setting/:id', component: UserSetting }

这是一个好方法,但并不是唯一的方法。

对于本例来说,用户中心页面的用户设置页面所需要参数的类型有明显的差异,用户编号必需是数值,用户则不能是纯数字。因此,可以通过正则约束来实现不同类型的参数匹配到相应的路由组件。

{ path: '/user/:username', component: User },

{ path: '/user/:id(\\d+)', component: UserSetting }

结果:

3、路由的嵌套

渲染路由的地方只有一个,即只有一个<router-view> </router-view>出口,这类路由实际上都是顶级路由。在实际开发中,我们的项目可能非常复杂,除了根组件中需要路由外,一些子组件中可能也需要路由,vue router提供了嵌套路由技术来支持这类场景。

在components文件夹下新建一个名为Friends.vue的文件。

<template><h1>好友列表</h1><h1>好友人数:{{$route.params.count}}</h1>
</template>

Friends组件只会在用户中心使用,可以将其作为一个子路由进行定义。

在User.vue中添加<router-view> </router-view>出口

<RouterView></RouterView>

User组件本身也是由路由管理的,在User组件内部再使用的 <router-view> </router-view>标签实际上定义的是二级路由的页面出口。

在main.js中定义二级路由

import Friends from './components/Friends.vue'
{path: '/user/:username', component: User, children: [{path: 'friends/:count',component: Friends}]},

在浏览器中输入:

http://localhost:3000/#/user/小王/friends/6

结果:

三、页面导航

1、使用路由的方法

当我们成功向vue应用注册路由后,在任何vue实例中,都可以通过$route属性访问路由对象。通过调用路由对象的push方法可以向history栈中添加一个新的记录。也就是说,用户可以通过浏览器的返回按钮返回上一个路由的URL。

修改App.vue文件代码如下

<el-button type="primary" @click="toUser">用户中心</el-button>
toUser() {this.$router.push({path: "/user/小王"})}

结果:

push方法可以接受一个对象,对象中通过path属性配置其URL路径。push方法也支持直接传入一个字符串作为URL路径

this.$router.push("/user/小王")

也可以通过路由名加参数的方式让vue router自动生成URL,要使用这种方法进行路由跳转,在定义路由的时候需要对路由进行命名

const routes=[

path:'/user/:username?',

name:'user',

component:User

]

this.$router.push({

name:'user',

params:{

username:'小王'

}

})

如果路由需要查询参数,可以通过query属性进行设置

this.$route.push({

path:'/user',

query:{

name:'xixi'

}

})

会被处理成/user?name=xixi

需要注意,在调用push方法配置路由对象时,如果设置了path属性,则params属性会被自动忽略,push方法本身也会返回一个promise对象,可以用其来处理路由跳转成功后的逻辑

this.$router.push({path: "/user/小王"}).then(() => {this.$message({message: '跳转成功',type: "success"})}, (reason) => { })

四、关于路由的命名

1、使用名称进行路由切换

router-link组件和push方法都可以根据名称进行路由切换。

this.$router.push({

name:'user',

params:{

username:'小王'

}

})

等价于

<router-link :to="{name:'user',params:{username:'小王'}}">小王</router-link>

2、路由视图命名

路由视图命名是指对router-view组件进行命名,router-view组件用来定义路由组件的出口,前面我们说过,路由支持嵌套,router-view可以进行嵌套。通过嵌套,允许我们的vue应用中出现读个router-view组件。但是对于有些场景,可能需要同级地展示多个路由视图,如顶部导航区和内容区两部分都需要使用路由组件,这时候就需要同级地使用router-view组件,要定义同级的每个router-view要展示的组件,可以对其进行命名。修改APP.vue文件。

<el-container><el-header><RouterView name="topBar"></RouterView></el-header><el-main><RouterView name="main"></RouterView></el-main><el-footer><RouterView></RouterView></el-footer></el-container>

修改main.js文件

{path: '/',components: {topBar: User,main: UserSetting,default: Demo1}}

结果:

3、使用别名

别名提供了一种路由路径映射的方式,也就是说我们可以自由地将组件映射到一个任意的路径上,而不用受到嵌套解构的限制。

{ path: '/user/:id(\\d+)', component: UserSetting, alias: '/setting/:id' },

两个路径的页面效果一模一样

http://localhost:3000/#/user/666

http://localhost:3000/#/setting/666

需要注意,别名和重定向并不完全一样,别名不会改变用户在浏览器中输入路径的本身,对于多级嵌套的路由来说,可以使用别名在路径上对其进行简化。如果员路由有参数配置,一定要注意别名也需要对应地包含这些参数。

const routes=[

{path:'/user/:id(\\d+)',component:UserSetting,alias:['/setting/:id','/s/:id']}

]

4、路由重定向

重定向也是通过路由配置来完成的,与别名的区别在于,重定向会将当前路由映射到另一个路由上,页面的URL会产生改变。

当用户访问路由/d时,会自动跳转到/demo1

{ path: '/d', redirect: '/demo1' },

redirect也支持配置为对象,设置对象的name属性,设置对象的name属性可以直接指定命名路由

{ path: '/d', redirect: { name: 'Demo' } },

{ path: '/demo1', component: Demo1, name: 'Demo' },

在实际开发中,更多时候会采用动态的方式配置重定向,对于需要用户登录才能访问的页面,为登录的用户访问此路由时,我们自动将其重定向到登录页面

{path: '/d', redirect: to => {//to是路由对象console.log(to);//随机数模拟登录状态let login = Math.random() > 0.5console.log(login);if (login) {return { path: '/demo1' }} else {return { path: '/demo2' }}}},

五、关于路由传参

在路由跳转时,可以通过参数的传递来进行后续的逻辑处理,在组件内部,之前使用$route.params的方式来获取路由传递的参数,这种方式虽然可行,但组件与路由紧紧地耦合在一起,并不利于组件的复用性。

由于之前在组件的模板内部使用了$route属性,这导致此组件的通用性大大降低,首先将其所有耦合路由的地方去掉

<template><h1>用户设置</h1><h2>id:{{id}}</h2>
</template>
<script>export default {props: ['id']}
</script>

现在,UserSetting组件能够通过外部传递的属性来做内部的逻辑,后面需要做的只是将路由的传参映射到外部属性上。vue router默认支持这一功能,路由配置方式如下:

{ path: '/user/:id(\\d+)', component: UserSetting, alias: '/setting/:id',props:true },

对于有多个页面出口的同级命名视图,我们需要对每个视图的props单独进行设置

{path: '/',components: {topBar: User,main: UserSetting,default: Demo1},props: { topBar: true, main: true, default: true }}

如果组件内部需要的参数与路由本身并没有直接关系,也可以将props设置为对象,此时props设置的数据将原样传递给组件的外部属性。

{ path: '/user/:id(\\d+)', component: UserSetting,props:{id:'000'}},

如上图所示,此时路由中的参数将被弃用,组件中获取到的id属性值将被固定为‘000’

props还有一种更强大的使用方式,可以直接将其设置为一个函数,函数中返回要传递到组件的外部属性对象,这种方式动态性很好

{ path: '/user/:id(\\d+)', component: UserSetting, alias: '/setting/:id', props:route=>{return{id:route.params.id,}} },

六、路由导航守卫

导航守卫的主要作用是在进行路由跳转时决定通过此次跳转或拒接此次跳转。

1、定义全局导航守卫

我们使用createRouter方法来创建路由实例,此路由实例可以使用beforeEach方法来注册全局的前置导航守卫,之后当触发导航跳转时,都会被此导航守卫捕获。

router.beforeEach((to, from) => {console.log(to);//将要跳转的路由对象console.log(from);//当前将要离开的路由对象return false//返回true表示允许此次跳转,返回false表示禁止此次跳转
})

当注册的beforeEach方法返回的是布尔值时,期用来决定是否允许此次导航跳转,以上代码表示,所有路由跳转都被禁止。

更多时候,我们会在beforeEach方法中返回一个路由配置对象来决定要跳转的页面,这种方式更加灵活,例如可以将登录态校验的逻辑放在全局守卫中处理

{path: '/user/:id(\\d+)', name: 'setting', component: UserSetting, alias: '/setting/:id', props: route => {return {id: route.params.id,}}},
router.beforeEach((to, from) => {console.log(to);//将要跳转的路由对象console.log(from);//当前将要离开的路由对象/* return false//返回true表示允许此次跳转,返回false表示禁止此次跳转 */if (to.name != 'setting') {return { name: 'setting', params: { id: '000' } }//返回要跳转到的2路由}
})

与定义全局前置守卫类似,也可以注册全局的导航后置回调。与前置守卫不同的是,后置回调不会改变导航本身,但是其对页面的分析和监控十分有用。

router.afterEach((to, from, failure) => {console.log('跳转结束');console.log(to);console.log(from);console.log(failure);
})

结果:

failure参数可以记录导航的异常信息。

2、为特定的路由注册注册导航守卫

如果只有特定的场景需要在页面跳转过程中实现相关逻辑,也可以为指定的路由注册导航守卫。有两种注册方式,一种是在配置路由时进行定义,另一种是在组件中定义。

path: '/demo1', component: Demo1, name: 'Demo', beforeEnter: (to, from, next) => {return false}

当用户访问/demo1路由对应的组件都会被拒接掉,需要注意,beforeEnter设置的守卫只有在进入路由时会触发,路由的参数变化并不会触发此守卫。

在编写组件时,也可以实现一些方法来为组件定制守卫函数

<template><h1>我是页面二</h1>
</template>
<script>export default {beforeRouteEnter: (to, from, next) => {console.log(to, from, '前置守卫');},beforeRouteUpdate: (to, from, next) => {console.log(to, from, '路由参数有更新时的守卫');},beforeRouteLeave: (to, from, next) => {console.log(to, from, '离开页面');},}
</script>

结果:

beforeRouteEnter是组件的导航前置守卫,在通过路由将要切换到当前组件时被调用,在这个函数中,可以做拦截操作,也可以做重定向操作,需要注意此方法只有在第一次切换此组件时会被调用,路由参数的变化不会重复调用此方法。beforeRouteUpdate方法在当前路由发生变化时会被调用,例如由参数的变化等都可以在此方法中捕获。beforeRouteLeave方法会在将要离开当前页面时被调用。还有一点特别需要注意 beforeRouteEnter方法中不能使用this来获取当前组件实例,因为在导航守卫确认通过前,新的组件还没有被创建,如果一定要在导航被确认时使用当前组件实例处理一些逻辑,可以通过next参数注册回调方法。

beforeRouteEnter(to, from, next) {console.log(to, from, '前置守卫');next((w) => {console.log(w);//w为当前组件实例})},

结果:

当前置守卫确认了此次跳转后,next参数注册的回调方法会被执行,并且会将当前组件的实例作为参数传入,在beforeRouteUpdate和beforeRouteLeave方法中可以直接使用this关键字来获取当前组件实例。

七、动态路由

1、动态添加与删除路由

在Vue Route中,动态操作路由的方法主要有两个:addRoute和removeRoute,addRoute用来动态添加一条路由,对应的removeRoute用来动态删除一条路由。

修改Demo1.vue

<template><h1>我是页面一</h1><el-button type="primary" @click="click">跳转demo2</el-button>
</template>
<script>import Demo2 from './Demo2.vue'export default {created() {console.log(this);this.$router.addRoute({path: '/demo2',component: Demo2})},methods: {click() {this.$router.push('/demo2')}}}
</script>

修改main.js文件

{path: '/demo1', component: Demo1, name: 'Demo', /* beforeEnter: (to, from, next) => {return false} */

结果:

以下这几种场景会触发路由的删除。

当使用addRoute方法动态添加路由时,如果添加了重名的路由,旧的会被删除

在调用addRoute方法时,他其实会返回一个删除回调,我们可以通过此删除回调来直接删除所添加的路由。

let call = this.$router.addRoute({path: '/demo2',component: Demo2,name: 'Demo2'})call()//删除此路由

对于已经命名的路由,可以通过名称来对路由进行删除

<template><h1>我是页面一</h1><el-button type="primary" @click="click">跳转demo2</el-button><el-button type="danger" @click="remove">删除demo2动态路由</el-button>
</template>
<script>import Demo2 from './Demo2.vue'export default {created() {console.log(this);this.$router.addRoute({path: '/demo2',component: Demo2,name: 'Demo2'})},methods: {click() {this.$router.push('/demo2')},remove() {this.$router.removeRoute('Demo2')}}}
</script>

结果:

没有删除前:

删除后:

路由被删除时,其所有的别名和子路由也会同步被删除

hasRouter方法用来检查当前已经注册的路由中是否包含某个路由,getRoutes方法用来获取包含所有路由的列表。

vue-vue路由管理相关推荐

  1. Vue中路由管理器Vue Router使用介绍(三)

    2019独角兽企业重金招聘Python工程师标准>>> 一.路由定义添加动态参数定义 1.路由定义项,使用:xx 方式 定义动态参数 {path:'/user/:id/:name', ...

  2. Vue Router 路由管理

    文章目录 Vue Router 路由管理 概述 安装 简单使用 定义2个组件 定义路由信息 支持路由 使用路由 动态路由 配置动态路由 配置404 限制动态参数 嵌套路由 命名路由 编程式导航 简单使 ...

  3. Vue Router路由管理器

    目录: 相关理解 基本路由 几个注意事项 嵌套(多级)路由 路由的query参数 命名路由 路由的params参数 路由的props配置 路由跳转的replace方法 编程式路由导航 缓存路由组件 a ...

  4. python vue token_Haytham个人博客开发日志 -- Flask+Vue基于token的登录状态与路由管理...

    指路牌 符合一下关键词,这篇博客有可能会对你有帮助 不使用工厂函数的Flask应用 不使用蓝本的Flask应用 Flask跨域配置 基于Token的登录状态管理 Flask+Vue Vue路由拦截 A ...

  5. Vue原理解析:手写路由管理器 —— Vue-Router

    由于时间问题,暂时先把代码完整的贴上来,感兴趣的朋友可以自行研究或收藏.等我那时有时间的时候,对专栏文章进行排序,并逐一讲解代码 一.对外暴露的入口文件index.js import Vue from ...

  6. vue的路由与es6的import, export

    vue主要用来实现前端模块化编程, 它的最终代码是一些序列化的js,简单的index.html访问入口,和一些image, vue的js使用es6来模块化设计, 为什么要这么做呢,我想主要是为了前端与 ...

  7. html5怎么改为vue_【面试需要Vue全家桶】一文带你看透Vue前端路由

    作者 |  Jeskson 掘金 |  https://juejin.im/user/5a16e1f3f265da43128096cb 2020.1.11 背景介绍 vue router是vue.js ...

  8. Vue前端路由(Vue-router)

    1.路由的概念 路由的本质就是一种对应关系,比如说我们在url地址中输入我们要访问的url地址之后,浏览器要去请求这个url地址对应的资源. 那么url地址和真实的资源之间就有一种对应的关系,就是路由 ...

  9. 【Vue】路由Router嵌套的实现及经典案例

    Vue 中路由在使用的时候 嵌套 使用是非常频繁的,所以本文我们就来通过案例介绍一下嵌套路由的使用. Vue路由嵌套 先来准备一个普通的路由案例: <!DOCTYPE html> < ...

  10. Vue前端路由~满满的干货

    文章目录 目标 目录 路由的基本概念与原理 路由 后端路由 SPA(Single Page Application) 前端路由 实现简易前端路由 Vue Router vue-router的基本使用 ...

最新文章

  1. java I/O总结(收藏)
  2. BZOJ 3261 最大异或和 可持久化Trie树
  3. AngularJS自定义表单验证
  4. (转)Unity3d UnityEditor编辑器定制和开发插件
  5. textview多行ellipsize=end不显示省略号的解决方法(一)
  6. linux系统下PHP无法调用oracle数据库的解决方法
  7. 浅析HTML5高薪的原因所在
  8. chrome不能安装adblock插件
  9. matlab hurst代码,Hurst指数法matlab程序求助
  10. SSM项目实例——简易版图书管理系统
  11. 14 款命令行常用工具的替代品
  12. Pytorch:optim.zero_grad()、pred=model(input)、loss=criterion(pred,tgt)、loss.backward()、optim.step()的作用
  13. 京东小程序开放平台,他来了
  14. python 'utf-8' codec can't decode byte 0xca in position 21: invalid continuation byte
  15. oracle序列无缓存,oracle 序列跳号现象
  16. 内码转换技术 (转)
  17. Enable tracemalloc to get the object allocation traceback及ResourceWarning: unclosed file
  18. spring 中的JMS是什么?
  19. 贝塞尔曲线公式,我是你爸爸
  20. 算白塞尔公式matlab,中误差计算公式

热门文章

  1. Vova and Trophies _Codeforces 1082B
  2. Arduino系列教程之 – PWM 的秘密(下)
  3. Samba实现windows/ubuntu 文件共享
  4. 搜狗浏览器如何设置截屏快捷键?
  5. 告别阳台储物,把它变成城市中舒适和安静的避难所
  6. 【算法入门11】反转链表
  7. 抖音SEO源码系统怎么搭建?
  8. Jmeter配置之线程组启动和配置说明
  9. 人体行为识别 动作属性
  10. volatile为什么不能保证原子性