Vue声明式导航 编程式导航、导航守卫、axios拦截器

news2025/1/17 0:20:39

一、声明式导航 & 编程式导航

       1. 声明式导航:以超链接方式实现的页面跳转,就是声明式导航

  • < a href=‘url’> 链接文本或图像 < /a >
  • < router-link to=‘url’ > 链接文本或图像 < /router-link >

       2. 编程式导航:通过javascript提供的api方法实现页面的跳转,就是编程式导航

  • location.href = ‘url’ 
  • location.go(number)
  • location.replace(‘url’)

       3. vue-router中提供的编程式导航的API

       ​ (1)this.$router.push(‘url’):将‘url’添加到路由表中,增加了一条路由记录

<template>
<div>
 <h3>Home 组件</h3>
 <button @click="gotoMovie">跳转到 Movie 页面</button>
</div>
</template>

<script>
export default {
  name: "",
  methods:{
    gotoMovie(){
      this.$router.push('/movie/1')
    }
  }
}
</script>

​        (2)this.$router.replace(‘url’):跳转到url中,并替换掉当前的历史记录

      ​          push 和 replace 的区别:
       ​        ​ 1)push 会增加一条历史记录
       ​        ​ 2)replace 不会增加历史记录,而是替换掉当前的历史记录

​        (3)this.$router.go(number):参数是一个数值,页面跳转到指定的位置,可以在浏览历史中前进和后退。

​ ​       (4)this.$router.go()的简化写法

​ ​     ​           1)this.$router.back(): 在历史记录中,后退到上一个页面

​        ​        ​ 2)this.$router.forword():在历史记录中,前进到下一个页面

<template>
<div>
 <h3>Home 组件</h3>
 <button @click="goBack">退回上一页</button>
</div>
</template>

<script>
export default {
  props:['id'],
  methods:{
    goBack(){
      this.$router.go(-1)
    }
  }
}
</script>

二、导航守卫

       1. 用途:在页面导航过程中实现重定向、取消路由、权限验证等业务。

​        导航守卫分为三类:全局守卫、路由独享的守卫、组件内守卫,可以用于路由导航过程中的不同阶段。

​        每一个导航守卫都有三个参数:to、fromnext (router、afterEach 除外)

​        2. 分类:全局守卫、组件内部守卫、路由独享的守卫

​        3. 全局守卫

​ ​        (1)全局前置守卫:每次发生路由的导航跳转时,都会触发全局前置守卫。因此,在全局前置守卫中,程序员可以对每个路由进行访问权限的控制。使用的router.beforeEach(to,from,next){ }来注册。当一个导航触发时,全局前置守卫按照路由创建的顺序调用。

  • to:将要访问的路由的信息对象
  • from:将要离开的路由的信息对象
  • ​next:是一个函数,调用next()表示当前路由已经放行

​        ​ (2)next调用的情况

​        ​ 1)用户拥有了权限,直接放行:next()

​        ​ 2)用户没有权限,强制跳转到指定的页面:next(‘/login’)

​        ​ 3)用户没有权限,不允许访问:next(false)

​        ​  4. 全局前置守卫的使用

​        ​ ​ (1)创建Login.vue组件

​        ​ ​ (2)在路由文件router / index.js中注册全局的前置守卫

//App.vue

<template>
  <div>
    <h2>{{ info }}</h2>
    <div>
      <label>
        账号:<input type="text" v-model.trim="userName" />
      </label>
      <br /><br />
      <label>
        密码:<input type="password" v-model.trim="passWord" />
      </label>
      <br /><br />
      <button type="button" @click="login">登录</button>
    </div>
  </div>
</template>

<script type="text/javascript">
import $http from '../axios/index'

export default {
  name: "Login",
  data() {
    return {
      info: '',
      userName: '',
      passWord: '',
    }
  },

  methods: {
    login() {
      //   //对登陆信息进行验证(实际中在此处进行ajax的请求)
      //   if('lisi' === this.userName && '1234' === this.passWord){
      //     sessionStorage.setItem('isAuth',true) //在页面缓存中保存isAuth,isAuth=true表示用户已登录
      //     this.info = ''

      //     //判断当前路由对象中参数是否有参数
      //     if(this.$route.query.redirect){
      //       let redirect =this.$route.query.redirect
      //       this.$router.replace(redirect)//跳转到指定页面
      //     }
      //     else{
      //       this.$router.replace('/')//若路由对象中没有redirect,则直接跳转到默认的首页
      //     }
      //   }else{//非法用户
      //   sessionStorage.setItem('isAuth',false)
      //   this.userName = ''
      //   this.passWord = ''
      //   this.info = '用户名或密码错误'
      // }
      $http.post('/test/login', {
        userName: this.userName,
        passWord: this.passWord
      }).then(res => {
        if (res.data.code == 200) {
          sessionStorage.setItem('auth', res.data.tokenInfo)
          this.info = ''
          //判断当前路由对象中参数是否有参数
          if (this.$route.query.redirect) {
            let redirect = this.$route.query.redirect
            this.$router.replace(redirect)//跳转到指定页面
          }
          else {
            this.$router.replace('/')//若路由对象中没有redirect,则直接跳转到默认的首页
          }
        } else {
          sessionStorage.setItem('auth', '')
          this.userName = ''
          this.passWord = ''
          this.info = '用户名或密码错误'
        }
      })
    }
  }
}
</script>

<style scoped>

</style>
//router/insex.js

//注册全局前置导航守卫
router.beforeEach((to, from, next) => {
    //判断目标路由是否是/login,如果是,则直接调用next()方法
    if (to.path == '/login') {
        next()
    } else {//否则判断用户是否已经登录,注意这里是字符串判断
        if (sessionStorage.getItem('auth')!='' && sessionStorage.getItem('auth'!=null)) {
            next()
        } else {//如果用户访问的是受保护的资源,且没有登录,则跳转到登录页面
            //并将当前路由的完整路径作为查询参数传递给Login组件,以便登录成功后返回先前的页面
            next({//强制跳转到登录页面
                path: '/login',
                query:{ redirect:to.fullPath}
            }
            )
        }

    }
})

三、axios拦截器

       1. axios模块的作用:是对基于http请求的封装。在浏览器对异步请求对象XMLHttpRequest进行封装

       2. 拦截器

       ​ (1)请求拦截器:对客户端发起的请求进行统一的前期处理(token、时间戳、cookie等)

       ​ (2)响应拦截器:对服务器端响应给客户端的数据统一进行处理之后再发给客户端

       3. 使用方法

//前端/App.vue

<template>
  <div id="app">
    <img alt="Vue logo" src="./assets/logo.png">
  <br/>  <br/>
  <button @click="login">登录</button>
  &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;
  <button @click="test">测试拦截器</button>
  </div>
</template>

<script>
import $http from './axios/index'
export default {
  name: 'App',
  methods:{
    login(){
      $http.post('/users/login',{
        userName:'lisi',
        userPwd:'12345'
      }).then(res=>{
        if(res.data.code == 200){
          sessionStorage.setItem('Auth',res.data.mytoken)
        }
      }).catch(err=>{
        console.log(err);
      })
    },
    test(){
      $http.post('/users/test').then(res=>{
        console.log(res.data);
      }).catch(err=>{
        console.log(err);
      })
    }
  }
}
</script>
//前端/index.js

import axios from "axios";
//1. 创建axios的实例,配置基础路径
const axiosInstance = axios.create({
    baseURL: 'http://localhost:8089',
    timeout: 5000
})

//2. 定义请求拦截器:给所有请求都带上token
axiosInstance.interceptors.request.use((req) => {
    let token = sessionStorage.getItem('Auth') //获取页面存储中的token信息
    if (token) { //若token存在
        req.headers['Auth'] = token
    }
    return req;
}, (err) => {
    return Promise.reject(err)
})

// 3.响应拦截器:对服务器响应给客户端的数据进行统一的处理
axiosInstance.interceptors.response.use((res) => {
    //1.对响应数据进行处理
    let result = res.data
    let code = result.code
    if (code == 200) {
        return result
    } else {
        return Promise.reject(result)
    }
}, (err) => {
    return Promise.reject(err)
})

export default axiosInstance
//后台/usrs.js

var express = require('express');
var router = express.Router();
var jwt = require('jsonwebtoken')

/* http://localhost:8089/users/login */
router.post('/login', (req, res)=>{
  //1.接收客户的请求数据
  let user = {
    name:req.body.userName,
    pwd:req.body.userPwd
  }
  //2.定义密钥
  let temp = 'baihualin'

  //3.生成token
  let token = jwt.sign(user,temp)

  res.json({
    code:200,
    mytoken:token
  })
} );

/* http://localhost:8089/users/login */
router.post('/test',(req,res)=>{
  //输出请求头信息
  console.log(req,headers)

  let arr = [
    {
      bookId:1001,
      bookName:'Vue2从入门到精通',
      publish:'清华大学出版社'
    }, 
    {
      bookId:1002,
      bookName:'Html从入门到放弃',
      publish:'清华大学出版社'
    }, 
    {
      bookId:1003,
      bookName:'Css都是浮云',
      publish:'清华大学出版社'
    }, 
  ]
  res.json(arr)
})


module.exports = router;

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/162118.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

录屏软件哪个好?电脑录屏软件排行榜推荐

你是不是还在为选不到合适的录屏软件而苦恼&#xff1f;市面上录屏软件种类繁多&#xff0c;功能参差不齐确实不好选择。录屏软件哪个好&#xff1f;怎样才能找到适合自己的录屏软件&#xff1f;不用焦虑。今天小编给大家推荐3款电脑录屏软件排行榜前列的录屏软件&#xff0c;每…

day37【代码随想录】贪心算法之划分字母区间、合并区间、单调递增的数字、买卖股票的最佳时机含手续费、监控二叉树

文章目录前言一、划分字母区间&#xff08;力扣763&#xff09;二、合并区间&#xff08;力扣56&#xff09;三、单调递增的数字&#xff08;力扣738&#xff09;四、买卖股票的最佳时机含手续费&#xff08;力扣714&#xff09;五、监控二叉树&#xff08;力扣968&#xff09;…

1.14 IIC总线实验

一.IIC总线&#xff1a; 1.同步半双工串行总线&#xff0c;用于同一个开发板两个芯片之间的通信 2.有两根信号线&#xff0c;一根SDA,一根SCL 3.IIC总线需要外接两个上拉电阻&#xff0c;使空闲状态保持高电平 4.IIC总线支持多主机多从机模式&#xff0c;一般采用单主机多从…

STM32外部中断解析

文章目录前言一、外部中断是什么二、STM32F103的外部中断三、外部中断的中断号四、HAL库的外部中断初始化流程总结前言 本篇文章将带大家了解STM32F103的外部中断。 一、外部中断是什么 外部中断是单片机实时地处理外部事件的一种内部机制。当某种外部事件发生时&#xff0c…

数组名的意义

数组名只有单独放在sizeof内部以及放在&后才代表整个数组的地址。其余情况数组名都表示数组首元素地址。 之前我们说过用sizeof(a)计算的是整个数组的大小&#xff0c;现在我们知道其中的原因了。由于sizeof里的数组名a表示整个数组的地址&#xff0c;故sizeof(a)求的是整…

Android10以上系统Audio音频遇到播放无声时的分析方法

​商务合作 2023年招聘 2023年逆向分析资料汇总 推荐阅读 Android Audio音频系统 Android Audio音频系统之深入浅出 Android Framework/驱动/内核中高级工程师 ​Android10以上系统Audio音频遇到播放视频无声时的分析方法 干货|Android APP应用工程师转Framework工程师(…

51单片机存储结构

之前概要介绍了8151微控制器的结构&#xff08;也就是51单片机&#xff09;。相比微处理器&#xff0c;微控制器的区别之一是在一个芯片上有程序存储器(RAM)和数据存储器(RAM)。存储区是微控制器非常重要的内容。 本文就介绍一下8051的存储结构。包括存储器的组织、处理器对存储…

VTK-Tessellator Subdivision

前言&#xff1a;本博文主要研究Tessellator 的Subdivision&#xff0c;对vtk中的所有相关接口进行研究&#xff0c;并找出最优的解决方法。 GeometricObjects中vtkTessellatorFilter的应用实例 待研究对当前的Subdivision进行优化。 vtkTessellatorFilter 位置&#xff1a;…

Docker容器数据卷与数据卷容器

文章目录什么是容器数据卷使用数据卷使用 docker 安装 mysql匿名和具名挂载数据卷容器什么是容器数据卷 Docker理念回顾 将应用和运行的环境打包形成容器运行&#xff0c;运行可以伴随着容器&#xff0c;但是我们对于数据的要求&#xff0c;是希望能够 持久化的&#xff01; 就…

【寒假每日一题】DAY.6 有序序列的合并

牛客网例题&#xff1a;点我做题 描述 输入两个升序排列的序列&#xff0c;将两个序列合并为一个有序序列并输出。数据范围&#xff1a; 1 ≤ n&#xff0c;m ≤ 1000&#xff0c;序列中的值满足0≤val≤30000输入描述&#xff1a; 输入包含三行&#xff0c;第一行包含两个正整…

微信小程序开发起步

一、小程序与普通网页开发的区别 1&#xff09;运行环境不同。网页运行在浏览器环境中&#xff0c;小程序运行在微信环境中。 2&#xff09;API不同。由于运行环境不同&#xff0c;所以小程序中无法调用DOM和BOM的API&#xff0c;但是小程序可以调用微信环境提供的各种API&am…

zabbix的安装部署,一看就会

目录 1、系统组成 2、采集模式 3、监控类型 4、相关术语 5、安装部署 1、系统组成 2、采集模式 监控系统数据采集的工作模式可以分为 被动模式&#xff08;从服务器端到客户端采集数据&#xff0c;对应的英文单词是pull&#xff09; 主动模式 &#xff08;客户端主动上报…

java基于springboot的新生报到系统app新生报到安卓app新生入学报到小程序加论文

简介 本项目主要是新生报到系统&#xff0c;包含的新生入学流程的功能&#xff1a;新生可以在app里提交预报到日期确认报到&#xff0c;查看自己的学费缴费记录&#xff0c;更改自己的银行卡号&#xff0c;查看课表&#xff0c;查看寝室&#xff0c;查看自己的专业班级等个人信…

CubeMX配置:定时器中断及串口配置

本文通过CubeMX配置F103RCT6的定时器中断和串口。 文章目录1 引入2 配置过程2.1 配置时钟过程2.2 配置串口过程2.3 定时器中断配置3 生成工程代码4 测试串口及定时器中断周期5 总结1 引入 拿到一个STM32控制器&#xff0c;在啥也没有的情况下从零开始写软件&#xff0c;博主习…

Java K 个一组翻转链表

K 个一组翻转链表困难给你链表的头节点 head &#xff0c;每 k 个节点一组进行翻转&#xff0c;请你返回修改后的链表。k 是一个正整数&#xff0c;它的值小于或等于链表的长度。如果节点总数不是 k 的整数倍&#xff0c;那么请将最后剩余的节点保持原有顺序。你不能只是单纯的…

电视为何降价至200元依然没人买?爱奇艺给出了答案

近日爱奇艺提高降低会员的服务内容&#xff0c;消费者要享受到将内容投放到电视就要买更贵的会员服务&#xff0c;引发热议&#xff0c;此举或许能解释国内消费者为何如今不买电视了。2022年电视行业可谓是一个相当惨淡的行业&#xff0c;量价齐跌&#xff0c;随着2022年底的到…

计算机视觉OpenCv学习系列:第二部分、图像操作

第二部分、图像操作第一节、图像读取与显示1.图像理解2.图像读取与显示3.代码练习与测试第二节、图像色彩空间转换1.图像色彩空间2.函数说明与解释3.代码练习与测试第三节、图像对象的创建与赋值1.图像对象属性2.图像对象创建与赋值3.代码练习与测试第四节、图像像素的读写操作…

SpringBoot——关于controller参数校验

参数校验主要使用两个标签Validated和Valid&#xff1b; Valid是Hibernate的注解校验&#xff0c;Validated是spring的&#xff0c;是Valid的增强&#xff1b;这两个标签也有一些不同之处&#xff0c;Valid可以标注在成员属性上也可以嵌套校验&#xff0c;而Validated不行&…

Node.js 操作MySQL数据库

在讲Node.js中引入mysql模块之前先讲一下关于MySQL数据库&#xff0c;首先是关于MySQL数据库的安装和mysql服务以及对mysql命令和可视化工具的一个基本使用&#xff1b;那么在这里已经准备好了关于MySQL数据库的内容了&#xff1a; MySQL 数据库安装详细 &#xff1a; MySQL数…

redis主从复制,一主多仆启动流程

文章目录一、安装redis二、复制修改配置文件三、添加conf配置文件四、开启主从复制五、薪火相传六、反客为主7、哨兵模式一、安装redis redis安装笔记 https://blog.csdn.net/G_GUi/article/details/128361131 二、复制修改配置文件 把redis.conf复制到一个新建文件夹里面&a…