vue3 动态路由及使用动态路由后刷新界面出现空白页或者404

最近编写vue3动态路由的功能遇到了一些问题,处理好了,总结出来,希望能帮助到你。
  • 正片开始
    先写好本地缓存菜单的方法(存储、删除、获取)
// utils/menu.jsconst getMenuList = () => {return JSON.parse(localStorage.getItem('menu_list'));
};const setMenuList = (menuList) => {localStorage.setItem('menu_list', JSON.stringify(menuList));
};const clearMenuList = () => {localStorage.removeItem('menu_list');
};export { getMenuList, setMenuList, clearMenuList };
  • pinia (vuex) 里创建一个全局变量存储菜单数组 && 在 actions方法里创建登录和退出登录的异步方法
// store/modules/user.jsimport { defineStore } from "pinia";
import { setMenuList, clearMenuList, getMenuList } from "@/utils/menu";const useUserStore = defineStore("user", {state: () => {return {token: undefined,userMenuPermission: [] // 菜单数组初始我们设为空数组,等下需要他做判断}},getters: {userInfo(state) {return { ...state };},},actions: {// 设置用户信息setInfo(partial) {this.$patch(partial);},// 重置用户信息resetInfo() {this.$reset();},// 获取用户信息info() {// 这可判断一下 本地存储的菜单是否存在,存在就给 userMenuPermission赋值就行不掉接口,我这没有整,你们按照自己需求来 getUserInfo().then((res) => {this.setInfo(res.data); // 将用户信息设置给state,但是注意需要key相同setMenuList(res.data.userMenuPermission) // 缓存菜单数组generateRoutes(this.userMenuPermission) // 添加动态路由});},// 账号密码登录async loginPwd(form, pubKey) {// ...省略await setToken(res.data.token); // 保存token 存储手法和菜单的一样},// 退出登录logout() {userLogin.logout().then((res) => {this.logoutCallBack();});},// 退出登录清空token和用户信息logoutCallBack() {this.resetInfo(); // 清空 stateclearToken(); // 清除 tokenclearMenuList(); // 清除 MenuList},},
});
export default useUserStore; // 暴露模块
// store/index.js
import { createPinia } from "pinia";
import useUserStore from "./modules/user";const pinia = createPinia();export { useUserStore };
export default pinia;
  • 处理路由配置
// router/modules/dynamicRouter.js 处理菜单数组和动态添加路由import { DEFAULT_LAYOUT } from "./staticRouter"; // layout (布局)
import router from "@/router/index"; // 1. 引入views下的所有文件 (使用vite创建的需要这样动态导入地址)
const modules = import.meta.glob('@/views/**/*.vue')/*** router.addRoute(route: RouteRecord):动态添加路由* router.removeRoute(name: string | symbol):动态删除路由* router.hasRoute(name: string | symbol): 判断路由是否存在* router.getRoutes(): 获取路由列表*/// 根据菜单信息递归生成路由(记得递归一定要出口)
export function generateRoutes(menuList = []) {const routes = [];menuList.forEach((menu) => {let routeItem = {};//如果是目录if (menu.meta.type == 1) {routeItem = {//路由名称name: menu.name,//路由地址path: menu.path,//组件路径component: DEFAULT_LAYOUT, // 是目录就为布局路径children: [],meta: menu.meta};} else {//如果是菜单routeItem = {//路由名称name: menu.name,//路由地址path: menu.path,//组件路径component: modules[`/src/views${menu.component}/index.vue`], // 为菜单的地址就要根据后端返回的地址来拼接,`/src/views${menu.component}/index.vue`这段不是固定的哈要根据你的文件格式来自定义,否则在上面导出文件数组中匹配不到会返回 undefinedmeta: menu.meta,children: [],};}if (menu.children && menu.children.length > 0) {routeItem.children = generateRoutes(menu.children);}if (!router.hasRoute(routeItem.name)) { // 防止添加了相同的菜单router.addRoute('layout', routeItem)}routes.push(routeItem); // 这个是拼接好处理的路由数组});return routes; // 根据需要返回即可
}

上面可能会遇到的问题:
modules[/src/views${menu.component}/index.vue] 会匹配不到你后端返回的路径,你要检查一下拼接格式是否正确,可打印出来和 const modules = import.meta.glob(‘@/views/**/*.vue’)数组里的对比查看问题

// router/modules/staticRouter.js 放置一些静态路由资源比如404、login等页面/*** * layout (布局)*/
export const DEFAULT_LAYOUT = () => import('@/layout/default-layout.vue');/*** staticRouter (静态路由)*/
export const staticRouter = [{path: '/',redirect: '/dashboard',},{path: '/login',name: 'login',component: () => import('@/views/login/index.vue'),meta: {requiresAuth: false,},},
];export const staticMenuRouter = [{name: "dashboard",path: "/dashboard",component: DEFAULT_LAYOUT,redirect: "/dashboard/index",meta: {title: "首页",sort: 1,icon: "icon-dashboard",},children: [{path: "index",name: "index",component: () => import("@/views/dashboard/index.vue"),meta: {title: "工作台",affix: true, // 用于判断TagsView是否一直固定hidden: true},},],}
]/*** errorRouter (错误页面路由)*/
export const errorRouter = [{path: "/403",name: "403",component: () => import("@/views/exception/403.vue"),meta: {title: "403页面"}},{path: "/404",name: "404",component: () => import("@/views/exception/404.vue"),meta: {title: "404页面"}},{path: "/500",name: "500",component: () => import("@/views/exception/500.vue"),meta: {title: "500页面"}},// 解决刷新页面,路由警告{path: "/:pathMatch(.*)*",component: () => import("@/views/exception/404.vue")}
];

上面可能会遇到的问题:
没有设置 path: “/:pathMatch(.)” 当刷新界面时,动态添加的路由会清空,地址栏的路由路径找不到就会报在这里插入图片描述不影响代码正常跑但是不好看。

// routerindex.js 配置路由模式,设置路由拦截等等import { createRouter, createWebHistory, createWebHashHistory } from 'vue-router';
import { staticRouter, errorRouter, staticMenuRouter } from "@/router/modules/staticRouter";
import { useUserStore } from '@/store';
import { isLogin } from '@/utils/cache/auth'; // 判断token是否存在// const userStore = useUserStore();const title = 'xxxxxx系统'// 白名单
let whiteList = ["login", '404'];export function getPageTitle(pageTitle) { // 拼接每个路由页面的名称显示在浏览器if (pageTitle) {return `${pageTitle} - ${title}`}return `${title}`
}// 设置路由模式,导入静态的路由
const router = createRouter({history: createWebHashHistory(),routes: [...staticRouter, ...errorRouter, ...staticMenuRouter],strict: false, // strict 为 true 时,将不会匹配带有尾部斜线的路由// sensitive: false, // 为 true 时,将区分大小写scrollBehavior: () => ({ left: 0, top: 0 }) // 设置页面滚动行为(始终滚动到顶部)
});// 路由前置首位
router.beforeEach(async (to, from, next) => {const userStore = useUserStore();// 1.NProgress 开始// NProgress.start(); // 一个加载动画,需要的朋友可以去看咋使用的 npm install --save nprogress// 2.判断是访问登陆页,有 Token 就在当前页面,没有 Token 重置路由到登陆页if (to.path.toLocaleLowerCase() === '/login') {if (isLogin()) return next(from.fullPath);resetRouter(); // 没有token 就重置路由return next(); // 放行去登录页}// 3.判断访问页面是否在路由白名单地址(静态路由)中,如果存在直接放行if (whiteList.includes(to.path)) return next();// 4.判断是否有 Token,没有重定向到 login 页面if (!isLogin()) return next({ path: '/login', replace: true });// 5.我们判断 vuex里面的设置的菜单列表是否为空,为空就重新获取列表添加路由if (!userStore.userMenuPermission.length) {await userStore.info();return next({ ...to, replace: true }); // 在已有的路由里自己匹配,replace: true 允许用户手贱点回退}// 8.正常访问页面next();});// 重置路由
const resetRouter = () => {const userStore = useUserStore();userStore.userMenuPermission.forEach(route => {const { name } = route;if (name && router.hasRoute(name)) router.removeRoute(name);});
};// 全局后置守卫
router.afterEach((to, from) => {// 动态设置浏览器标题document.title = getPageTitle(to.meta.title)// NProgress.done();
})export default router;

上面可能会遇到的问题:
刷新界面后重新获取路由动态添加路由,要是异步的
在这里插入图片描述
否则会出现 next({ …to, replace: true }) 一直匹配不到存在的路由死循环
在这里插入图片描述

  • 到这就差不多了,因该可以用了,啰嗦了点

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

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

相关文章

MachineSink - 优化阅读笔记

注:该优化与全局子表达式消除刚好是相反的过程,具体该不该做这个优化得看代价模型算出来的结果(有采样文件指导算得会更准确) 该优化过程将指令移动到后继基本块中,以便它们不会在不需要其结果的路径上执行。 该优化过程并非旨在替代或完全…

【大厂AI课学习笔记NO.80】深度学习行业人才能力图谱

深度学习领域的就业岗位及所需关键技术、工具、能力分析 深度学习作为人工智能领域的一个重要分支,近年来得到了飞速的发展。随着技术的不断进步和应用场景的不断拓展,深度学习领域的就业岗位也日益增多。本文将从领军人才、产业研发人才、应用开发人才…

Linux——信号

目录 什么是信号 Linux下的信号 信号的记录 信号处理的常见方式 产生信号 使用组合键产生信号(包含core dump) 使用系统调用向进程发送信号 由软件条件产生信号 由硬件异常产生信号 阻塞信号 内核表示 sigset_t 信号集操作函数 sigpendin…

jvm八股

文章目录 运行时数据区域Java堆对象创建对象的内存布局对象的访问定位句柄直接指针 GC判断对象是否已死引用计数算法可达性分析算法 引用的类别垃圾收集算法分代收集理论标记清除算法标记复制算法标记整理算法 实现细节并发的可达性分析 垃圾收集器serial收集器ParNew收集器Par…

如何保证 redis 的高并发和高可用?redis 的主从复制原理能介绍一下么?redis 的哨兵原理能介绍一下么?

目录 一、面试官心理分析 二、面试题剖析 1.Redis 主从架构 2.Redis replication 的核心机制 3.Redis 主从复制的核心原理 4.主从复制的断点续传 5.无磁盘化复制 6.过期 key 处理 7.复制的完整流程 8.全量复制 9.增量复制 10.heartbeat 11.异步复制 12.Redis 如何…

【三十】springboot项目上高并发解决示例

互相交流入口地址 整体目录: 【一】springboot整合swagger 【二】springboot整合自定义swagger 【三】springboot整合token 【四】springboot整合mybatis-plus 【五】springboot整合mybatis-plus 【六】springboot整合redis 【七】springboot整合AOP实现日志操作 【…

Java学习记录(十六):IO流(三)

缓冲流 缓冲流是一种高级流,可以理解为将基本流包装成了这种高级流,而这种高级流的特点就是将一个缓冲区放入到基本流当中,从而提高基本流的效率,但实际进行读写操作的还是基本流 下面为缓冲字节输出流和缓冲字节输入流的应用&am…

基于SpringBoot疫情打卡健康评测系统

基于SpringBoot疫情打卡健康评测系统~ 开发语言:Java 数据库:MySQL 技术:SpringBootMyBatis 工具:IDEA/Ecilpse、Navicat、Maven 系统展示 学生端功能效果展示 试卷表 在线考试 打卡管理 居家管理 学生返校申请管理 管理…

救命!我终于会写一份简单的产品说明书了

写一份简单的说明书其实也不简单,需要有前期的调研准备,撰写时的条框梳理,收尾的清晰明了。网络上有很多撰写产品说明书的方法,却也有一些模糊的地方,下面就来看看LookLook同学是怎么完成一份简单的产品说明书的。 一、…

【Python】一文详细介绍plt.rcParams 在 Matplotlib 中的原理、作用、注意事项

【Python】一文详细介绍plt.rcParams 在 Matplotlib 中的原理、作用、注意事项 🌈 个人主页:高斯小哥 🔥 高质量专栏:Matplotlib之旅:零基础精通数据可视化、Python基础【高质量合集】、PyTorch零基础入门教程&#x…

c++ primer plus 笔记 第十六章 string类和标准模板库

string类 string自动调整大小的功能: string字符串是怎么占用内存空间的? 前景: 如果只给string字符串分配string字符串大小的空间,当一个string字符串附加到另一个string字符串上,这个string字符串是以占用…

css 用flex做成田字型

哈喽&#xff0c;各位小伙伴&#xff01;今天给大家来css控制div完成田字型样式&#xff0c;来&#xff0c;看看下面的效果图&#xff1a; 一看就知道你们想要代码了&#xff0c;不急。代码&#xff1a; <!DOCTYPE html> <html lang"en"> <head>&…

JAVA全面基础知识(第七部分)

大家好我是程序员阿存&#xff0c;混迹在java圈的辛苦码农。今天要和大家聊的是一款&#xff0c;项目源码以及部署相关请联系存哥&#xff0c;文末附上联系信息 。 这篇文章给大家分享的是JAVA的基础知识&#xff0c; &#x1f495;&#x1f495;作者&#xff1a;程序员阿存 &…

哪里下载短视频素材?推荐几个短视频素材下载网站

当短视频行业的迅速崛起&#xff0c;剪辑影视短片的魅力无法抗拒&#xff0c;越来越多朋友爱看短视频&#xff0c;但从哪里找到高清、无水印和无字幕的短视频素材呢&#xff1f;今天&#xff0c;我将为大家推荐几个可获取短视频素材的优秀网站&#xff0c;下面让我们一起去看看…

吴恩达机器学习-可选实验室:逻辑回归(Logistic Regression))

在这个不评分的实验中&#xff0c;你会探索sigmoid函数(也称为逻辑函数)探索逻辑回归;哪个用到了sigmoid函数 import numpy as np %matplotlib widget import matplotlib.pyplot as plt from plt_one_addpt_onclick import plt_one_addpt_onclick from lab_utils_common impor…

2024最新版使用PyCharm搭建Anaconda

2024最新版使用PyCharm搭建Anaconda 因为pycharm自带的包不全&#xff0c;或者下载的时候比较慢&#xff0c;所以我们直接用anaconda的包&#xff0c;毕竟我们以后还会学到很多的包&#xff0c;不多说&#xff0c;直接开干&#xff01; 一、下载Pycharm、Anacoda pycharm中文网…

信息系统三级等保安全解决方案(46页Word)

1、系统定级与安全域 2、实施方案设计 3、安全防护体系建设规划 软件开发全系资料分享下载&#xff1a;软件项目开发全套文档下载_软件开发文档下载-CSDN博客

安全防御第七次作业

拓扑图如图所示&#xff1a; 问题&#xff1a;在FW7和FW8之间建立一条IPSEC通道保证10.0.2.0/24网段 可以正常访问到192.168.1.0/24 注&#xff1a;基础配置我在此省略了 一、NAT配置 FW4&#xff1a; FW6&#xff1a; 二、在FW4上做服务器映射 三、配置IPSEC FW5&#xff…

MessAuto-让验证码提取更加丝滑

专注于web漏洞挖掘、内网渗透、免杀和代码审计&#xff0c;感谢各位师傅的关注&#xff01;网安之路漫长&#xff0c;与君共勉&#xff01; MessAuto MessAuto 是一款 macOS 平台自动提取短信和邮箱验证码到粘贴板的软件&#xff0c;由Rust开发&#xff0c;适用于任何APP 下面展…

这个简历你给几分?

求大家给给建议&#x1f62d;&#x1f62d; 自己算法几乎没刷过&#xff0c;项目也不咋好 打算21号开始投日常实习 好慌