当前位置: 首页 > news >正文

uniapp封装路由管理(兼容Vue2和Vue3)

1:uniapp已经有路由管理了为什么还要二次封装路由?

  • 简化配置和调用
  • 增强灵活性和可扩展性
  • 实现统一的功能和策略
  • 提升开发效率和团队协作

2. 增强灵活性和可扩展性

  • 灵活配置:二次封装允许开发者根据实际需求灵活配置路由参数,如跳转类型、页面参数等。这种灵活性使得路由管理更加符合项目的具体需求。
  • 易于扩展:随着项目的不断发展,可能需要新增或修改路由配置。通过二次封装,可以方便地扩展路由功能,无需对原有代码进行大规模修改。

2. 集中管理,简化配置

  • 统一配置:通过封装路由管理,将所有页面路由信息集中在一个地方进行配置和管理。这样不仅使得路由信息一目了然,还方便进行统一修改和维护。
  • 简化跳转:开发者只需调用封装好的路由方法,如$openPage,而无需关心底层的路由实现细节,大大简化了页面跳转的代码。

3. 提高代码的可读性和可维护性

  • 清晰的代码结构:通过封装,将路由处理逻辑与业务逻辑分离,使得代码结构更加清晰。开发者可以更容易地理解和维护项目的导航流程。
  • 易于扩展:当需要新增或修改路由时,只需在配置中进行调整,无需修改业务代码,降低了代码耦合度,提高了可维护性。

4. 增强代码的复用性和灵活性

  • 复用性:封装的路由方法可以在项目的不同部分重复使用,无需重复编写相同的跳转逻辑。
  • 灵活性:通过配置参数,如跳转类型(navigateToswitchTab等),可以灵活地实现不同的跳转需求,满足不同场景下的导航要求。

5. 便于实现统一的功能和策略

  • 统一处理:在路由封装中,可以方便地实现统一的路由拦截、权限校验等功能,确保所有页面跳转都遵循相同的规则。
  • 全局参数传递:通过封装,可以更方便地在页面间传递参数,实现全局状态管理,提升用户体验。

6. 提升开发效率和用户体验

  • 开发效率:封装的路由管理减少了重复代码的编写,提高了开发效率。开发者可以更快地实现页面跳转功能,专注于业务逻辑的开发。
  • 用户体验:通过统一的路由管理,可以确保页面跳转的一致性和流畅性,提升用户在应用中的导航体验。

统一管理的优势

  • 错误检查:在处理路由配置时,添加了错误检查机制,确保每个页面都有必要的属性(如name),从而避免运行时错误。
  • 默认值处理:在不需要指定跳转类型时,提供默认的navigateTo,简化了配置。
  • 支持SubPackages:通过处理subPackages,支持分包加载,优化了应用的性能和加载时间。

我这里简单做了一下封装有什么问题欢迎留言指正

在项目中创建min.router.config.js文件
/*** min.router.config.js* 使用方法* Vue2* 示例:this.$openPage({name: 'my', query: {id: 123}, type: 'navigateTo'})		传参方式* 示例:this.$openPage('my')    不传参数可以简写* Vue3 setup语法糖* 示例:* import { getCurrentInstance } from 'vue';* const { $openPage } = getCurrentInstance().appContext.config.globalProperties;* $openPage({name: 'my', query: {id: 123}, type: 'navigateTo'})		传参方式* $openPage('my')    不传参数可以简写* Vue2 和 Vue3* 视图中可直接使用 $openPage('my')* @param name			字符串地址名* @param query			传递的参数* @param type			跳转的方式 ['navigateTo', 'switchTab', 'reLaunch', 'redirectTo']  不传默认navigateTo* this.$parseURL()		解析路由参数* 使用方法在接收参数页this.$parseURL(options)*/
// #ifdef VUE3
import { getCurrentInstance } from 'vue';
// #endifconst toString = Object.prototype.toStringfunction isObject(value) {return toString.call(value) === '[object Object]'
}function isString(value) {return toString.call(value) === '[object String]'
}function isDefault(value) {return value === void 0
}function openPage(args) {let name, query = {}, queryStr = null, path, type, isName = false;switch (true) {case isObject(args):({ name, type, query = {} } = args)breakcase isString(args):name = args;breakdefault:throw new Error('参数必须是对象或者字符串')}if (isObject(query)) {queryStr = encodeURIComponent(JSON.stringify(query))} else {throw new Error('query数据必须是Object')}this.$routerConfig.forEach(item => {if (item.name === name) {path = item.path;type = type || (item.type || 'navigateTo');isName = true;}})if (!isName) {throw new Error(`没有${name}页面`)}if (!['navigateTo', 'switchTab', 'reLaunch', 'redirectTo'].includes(type)) {throw new Error(`name:${name}里面的type必须是以下的值['navigateTo', 'switchTab', 'reLaunch', 'redirectTo']`)}return new Promise((resolve, reject) => {uni[type]({url: `/${path}?query=${queryStr}`,success: resolve,fail: reject})})
}function parseURL() {// #ifdef VUE2const query = this.$root.$mp.query.query;// #endif// #ifdef VUE3const query = getCurrentInstance().proxy.$root.$scope.options.query;// #endifif (query) {return JSON.parse(decodeURIComponent(query))} else {return {}}
}function install(Vue, routerConfig) {if (!routerConfig || !Array.isArray(routerConfig.options.routes)) {throw new Error("routerConfig 必须是一个带有 routes 属性的对象");}// Vue 2 和 Vue 3 的兼容混入Vue.mixin({beforeCreate: function() {if (!isDefault(routerConfig)) {Vue._routerConfig = routerConfig;}}})// 定义全局属性const globalProperties =// #ifdef VUE2Vue.prototype// #endif// #ifdef VUE3Vue.config.globalProperties// #endif;Object.defineProperty(globalProperties, '$routerConfig', {get: function() {return Vue._routerConfig._router;}})Object.defineProperty(globalProperties, '$parseURL', {get: function() {return parseURL.bind(this);}})Object.defineProperty(globalProperties, '$openPage', {get: function() {return openPage.bind(this);}})
}function minRouterConfig(options) {if (!(this instanceof minRouterConfig)) {throw Error("minRouterConfig是一个构造函数,应该用`new`关键字调用")}isDefault(options) && (options = {})this.options = options;this._router = options.routes || [];
}minRouterConfig.install = install;
minRouterConfig.prototype.openPage = openPage;
minRouterConfig.prototype.parseURL = parseURL;
export default minRouterConfig;
创建router.config.js文件
/** 使用方法* Vue2* 示例:this.$openPage({name: 'my', query: {id: 123}, type: 'navigateTo'})		传参方式* 示例:this.$openPage('my')    不传参数可以简写 * 示例:<view @click="$openPage('ceshi')"></view>  奈何小程序不支持这样跳转* Vue3 setup语法糖* 示例:* import { getCurrentInstance } from 'vue';* const { $openPage } = getCurrentInstance().appContext.config.globalProperties;* $openPage({name: 'my', query: {id: 123}, type: 'navigateTo'})		传参方式* $openPage('my')    不传参数可以简写* Vue2 和 Vue3* 视图中可直接使用 $openPage('my')* 配置参数项说明:* name:可选配置 (路由名称)* path:必填配置 (路由地址)* type 跳转的方式 ['navigateTo', 'switchTab', 'reLaunch', 'redirectTo']  不传默认navigateTo*/
import pagesJson from '@/pages.json'
import minRouterConfig from './min.router.config.js'// 处理 pages 数组
function processPages(pages) {return pages.map(page => {if (!page.name) {throw Error(`Page is missing the 'name' property: ${JSON.stringify(page)}`);}const processedPage = {name: page.name,path: page.path}// 仅在 type 不存在时不添加默认值if(page.type) {processedPage.type = page.type;}return processedPage});
}// 处理 subPackages 数组
function processSubPackages(subPackages) {if(!subPackages) return []let routes = [];subPackages.forEach(subPackage => {routes = routes.concat(processPages(subPackage.pages).map(page => {return {...page,path: subPackage.root + '/' + page.path};}));});return routes;
}const router = new minRouterConfig({routes: [ // 权限路由 在main.js可实现路由拦截 所以路由都需要注册...processPages(pagesJson.pages),...processSubPackages(pagesJson.subPackages)]
})export default router

router.config.js文件无需配置任何路径,均在pages.json中配置即可

最后在main.js中我们全局挂载一下

import App from './App'// 引入全局方法
import minRouterConfig from '@/config/min.router.config.js'
import routerConfig from '@/config/router.config.js'// #ifndef VUE3
import Vue from 'vue'
import './uni.promisify.adaptor'
Vue.config.productionTip = false
App.mpType = 'app'
const app = new Vue({...App
})
app.$mount()
// #endif// #ifdef VUE3
import { createSSRApp } from 'vue'
export function createApp() {const app = createSSRApp(App)app.use(minRouterConfig, routerConfig)return {app}
}
// #endif

我这里是在项目根目录创建了一个config文件夹,两个文件都在config文件夹中

以上配置在uniapp框架下不管是Vue2或者是Vue3都通用

不管是Vue2的组合式API下还是Vue3的setup语法糖下都可以正常使用

在页面中使用

Vue3中 setup语法糖

<template><button @click="onClick">跳转我的</button>
</template><script setup>import { getCurrentInstance } from 'vue';// data数据const { $openPage } = getCurrentInstance().appContext.config.globalProperties;// methods方法const onClick = () => {$openPage('user')}
</script><style></style>

Vue2 组合式API

<template><button @click="onClick">跳转我的</button>
</template><script>export default {methods: {onClick() {this.$openPage('user')}}}
</script><style></style>

通用

<template><button @click="$openPage('user')">跳转我的</button>
</template><script></script><style></style>

以上有什么问题可以下面留言一起学习

相关文章:

uniapp封装路由管理(兼容Vue2和Vue3)

1&#xff1a;uniapp已经有路由管理了为什么还要二次封装路由&#xff1f; 简化配置和调用增强灵活性和可扩展性实现统一的功能和策略提升开发效率和团队协作 2. 增强灵活性和可扩展性 灵活配置&#xff1a;二次封装允许开发者根据实际需求灵活配置路由参数&#xff0c;如跳…...

π0源码解析——一个模型控制7种机械臂:对开源VLA sota之π0源码的全面分析,含我司的部分落地实践

前言 ChatGPT出来后的两年多&#xff0c;也是我疯狂写博的两年多(年初deepseek更引爆了下)&#xff0c;比如从创业起步时的15年到后来22年之间 每年2-6篇的&#xff0c;干到了23年30篇、24年65篇、25年前两月18篇&#xff0c;成了我在大模型和具身的原始技术积累 如今一转眼…...

【C++】Class(1)

《C程序设计基础教程》——刘厚泉&#xff0c;李政伟&#xff0c;二零一三年九月版&#xff0c;学习笔记 文章目录 1、类的定义1.1、结构体和类1.2、基本概念1.3、成员函数的定义1.4、内联成员函数 2、对象2.1、对象的定义2.2、成员访问 3、构造函数3.1、构造函数的定义3.2、子…...

doris: Oracle

Apache Doris JDBC Catalog 支持通过标准 JDBC 接口连接 Oracle 数据库。本文档介绍如何配置 Oracle 数据库连接。 使用须知​ 要连接到 Oracle 数据库&#xff0c;您需要 Oracle 19c, 18c, 12c, 11g 或 10g。 Oracle 数据库的 JDBC 驱动程序&#xff0c;您可以从 Maven 仓库…...

Android14 OTA差分包升级报Package is for source build

制作好差分包&#xff0c;使用adb线刷模式验证ota升级&#xff0c;出现E:Package is for source build错误 使用adb方式验证 进入recovery模式 adb reboot recovery稍等一会界面会提示 Now send the package you want to apply to the device with "adb sidelaod <…...

双向选择排序算法

一 概述 双向选择排序(又称鸡尾酒选择排序)是选择排序的优化版本,核心改进在于每轮遍历同时确定未排序部分的最小值和最大值,分别交换到序列两端,从而减少遍历轮数。 二 时间复杂度 时间复杂度为(O(n^2)),但实际比较次数约为标准选择排序的 (1/2)。 三 C++实现代…...

Node.js setImmediate 教程

Node.js setImmediate 教程 简介 setImmediate() 是 Node.js 环境中的一个函数&#xff0c;用于安排一个回调函数在当前事件循环周期结束后立即执行。它提供了一种在当前操作完成后&#xff0c;但在任何 I/O 事件或定时器触发之前执行代码的方法。 基本用法 setImmediate((…...

MyBatis @Param 注解详解:多参数传递与正确使用方式

Param 注解主要用于 MyBatis 进行参数传递时给 SQL 语句中的参数 起别名&#xff0c;通常用于 多参数 方法&#xff0c;使参数在 XML Mapper 文件或注解 SQL 语句中更清晰易用。 1. 基本用法 在 Mapper 接口中使用 Param 来为参数命名&#xff0c;避免 MyBatis 解析时出现参数…...

Spring实战spring-ai运行

目录 1. 配置 2 .搭建项目 3. 查看对应依赖 3.1 OpenAI 依赖 3.2 配置 OpenAI API 密钥 application.properties application.yml 4. openai实战 5. 运行和测试 6. 高级配置 示例&#xff1a;配置模型和参数 解释&#xff1a; 7. 处理异常和错误 示例&#xff1a;…...

STL:C++的超级工具箱(一)

书接上回&#xff0c;内存管理和指针&#xff1a;C的双刃手术刀&#xff08;一&#xff09;-CSDN博客&#xff0c;在上篇我们聊到了什么是内存&#xff0c;堆栈&#xff0c;内存管理和智能指针相关的内容&#xff0c;接下来让我们一起去看看STL是什么吧。 第一步&#xff1a;提…...

leetcode349 两个数组的交集

求两个数组的交集&#xff0c;直白点儿就是【nums2 的元素是否在 nums1 中】。 在一堆数中查找一个数&#xff0c;当然是扔出哈希。碰到这种对目前来说是未知数值大小的情况&#xff0c;我们可以使用集合 set 来解决。 使用数组来做哈希的题目&#xff0c;是因为题目都限制了数…...

快速生成viso流程图图片形式

我们在写详细设计文档的过程中总会不可避免的涉及到时序图或者流程图的绘制&#xff0c;viso这个软件大部分技术人员都会使用&#xff0c;但是想要画的好看&#xff0c;画的科学还是比较难的&#xff0c;现在我总结一套比较好的方法可以生成好看科学的viso图(图片格式)。主要思…...

鸿蒙Android4个脚有脚线

效果 min:number122max:number150Row(){Stack(){// 底Text().border({width:2,color:$r(app.color.yellow)}).height(this.max).aspectRatio(1)// 长Text().backgroundColor($r(app.color.white)).height(this.max).width(this.min)// 宽Text().backgroundColor($r(app.color.w…...

【NetTopologySuite类库】geojson和shp互转,和自定义对象互转

geojson介绍 1. 示例 在visual studio中使用NuGet中安装了三个库&#xff08;.net4.7.2环境&#xff09;&#xff1a; NetTopologySuite 2.5NetTopologySuite.IO.Esri.Shapefile 1.2NetTopologySuite.IO.GeoJSON 4.0 1.1 shp数据转geojson 先创建一个shp文件作为例子&…...

【哇! C++】类和对象(三) - 构造函数和析构函数

目录 一、构造函数 1.1 构造函数的引入 1.2 构造函数的定义和语法 1.2.1 无参构造函数&#xff1a; 1.2.2 带参构造函数 1.3 构造函数的特性 1.4 默认构造函数 二、析构函数 2.1 析构函数的概念 2.2 特性 如果一个类中什么成员都没有&#xff0c;简称为空类。 空类中…...

Ubuntu20.04本地配置IsaacLab 4.2.0的G1训练环境(一)

Ubuntu20.04本地配置IsaacLab的G1训练环境&#xff08;一&#xff09; 配置Omniverse环境配置IsaacSim配置IsaacLab 写在前面&#xff0c;如果Ubuntu剩余空间低于60G&#xff0c;则空间不足&#xff0c;除非你不需要资产包。但资产包中却包含了G1模型、Go2模型等机器人模型和代…...

浅谈汽车系统电压优缺点分析

汽车电气系统的电压等级选择直接影响整车性能、能效和兼容性。以下是 12V、24V、48V 系统的简单介绍&#xff0c;包括技术特点、优缺点及典型应用场景。 汽车电气系统的发展随着车辆电子设备的增多和对能效要求的提高&#xff0c;电压等级也在逐步提升&#xff0c;从传统的12V…...

Springboot基础篇(4):自动配置原理

1 自动配置原理剖析 1.1 加载配置类的源码追溯 自动配置的触发入口&#xff1a; SpringBootApplication 组合注解是自动配置的起点&#xff0c;其核心包含 EnableAutoConfiguration&#xff0c;该注解使用AutoConfigurationImportSelector 实现配置类的动态加载。 启动类的注…...

Dify 开源大语言模型应用开发平台使用(一)

文章目录 一、创建锂电池专业知识解答应用1.1 应用初始化二、核心功能模块详解2.1 知识库构建2.2 工作流与节点编排节点类型说明工作流设计示例:锂电池选型咨询2.3 变量管理三、测试与调试3.1 单元测试3.2 压力测试3.3 安全验证四、部署与优化建议4.1 部署配置4.2 持续优化结论…...

机器学习深度学习基本概念:logistic regression和softmax

逻辑回归用来处理二分类问题 softmax用来处理多分类问题&#xff1a;比如llm在generate的时候&#xff0c;每个batch里面的一个样本的一个一次generate就是softmax生成一个大小为vocab_size的向量的概率分布&#xff0c;然后再采样 逻辑回归&#xff08;logistic regression&…...

OpenCV计算摄影学(16)调整图像光照效果函数illuminationChange()

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 对选定区域内的梯度场应用适当的非线性变换&#xff0c;然后通过泊松求解器重新积分&#xff0c;可以局部修改图像的表观照明。 cv::illuminati…...

Git - 补充工作中常用的一些命令

Git - 补充工作中常用的一些命令 1 一些场景1.1 场景11.2 场景21.3 场景31.4 场景41.5 场景51.6 场景61.7 场景71.8 场景81.9 场景91.10 场景101.11 场景111.12 场景121.13 场景131.14 场景141.15 场景15 2 git cherry-pick \<commit-hash\> 和 git checkout branch \-\-…...

使用Python的requests库调用API并处理JSON响应的详细步骤

1. 安装request库 pip install requests 2. 发送GET请求 import requests# 定义API地址 url "https://api.example.com/data"# 发送GET请求 response requests.get(url)# 检查HTTP状态码 if response.status_code 200:# 解析JSON响应data response.json()prin…...

Mybatis如何通过databaseId属性支持不同数据库的不同语法

目录 一、前言 二、如何配置 三、源码解读 四、自定义 一、前言 在一次项目功能测试中&#xff0c;发现有个sql在其他嵌入式数据库中执行正常&#xff0c;但是在mysql中执行失败&#xff0c;发现是因为有个字段在mysql中是关键字&#xff0c;需要使用反引号&#xff08;&…...

android edittext 防止输入多个小数点或负号

有些英文系统的输入法,或者定制输入法。使用xml限制不了输入多个小数点和多个负号。所以代码来控制。 一、通过XML设置限制 <EditTextandroid:id="@+id/editTextNumber"android:layout_width="wrap_content"android:layout_height="wrap_conten…...

windows部署spleeter 版本2.4.0:分离音频的人声和背景音乐

windows部署spleeter 版本2.4.0&#xff1a;分离音频的人声和背景音乐 一、Spleeter 是什么&#xff1f; Spleeter 是由法国音乐流媒体公司 Deezer 开发并开源的一款基于深度学习的音频分离工具。它能够将音乐中的不同音轨&#xff08;如人声、鼓、贝斯、钢琴等&#xff09;分…...

深度学习、宽度学习、持续学习与终身学习:全面解析与其在大模型方面的应用

目录 引言&#xff1a; 1. 深度学习&#xff08;Deep Learning&#xff09; 1.1 深度学习的基本概念 1.2 深度学习的数学原理 1.3 深度学习的特点 1.4 深度学习在大模型中的应用 2. 宽度学习&#xff08;Wide Learning&#xff09; 2.1 宽度学习的基本概念 2.2宽度学习…...

【量化科普】Arbitrage,套利

【量化科普】Arbitrage&#xff0c;套利 &#x1f680;量化软件开通 &#x1f680;量化实战教程 什么是套利&#xff1f; 套利&#xff08;Arbitrage&#xff09;是金融市场中的一种交易策略&#xff0c;指的是在不同市场或不同形式中同时买入和卖出相同或相似的金融产品&a…...

删除已加入 .gitignore却仍被git追踪的文件

.gitignore 文件只会影响未被跟踪的文件&#xff0c;而已经被 Git 跟踪的文件不会因为被添加到 .gitignore 而停止被跟踪。 eg&#xff1a;例如在创建.gitignore文件前&#xff0c;已经将sync.sh文件推送到远程分支&#xff0c;因此该文件已被git追踪。 去掉sync.sh文件追踪的步…...

pytest框架 核心知识的系统复习

1. pytest 介绍 是什么&#xff1a;Python 最流行的单元测试框架之一&#xff0c;支持复杂的功能测试和插件扩展。 优点&#xff1a; 语法简洁&#xff08;用 assert 替代 self.assertEqual&#xff09;。 自动发现测试用例。 丰富的插件生态&#xff08;如失败重试、并发执…...