news 2026/6/10 14:10:57

[鸿蒙2025领航者闯关]HarmonyOS路由跳转

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
[鸿蒙2025领航者闯关]HarmonyOS路由跳转

问题描述

应用需要页面跳转和参数传递:

  • 点击按钮跳转到详情页
  • 携带参数跳转
  • 返回时传递结果
  • 路由拦截和鉴权

关键字:Router页面跳转参数传递路由导航

解决方案

1. 基础路由跳转

import { router } from '@kit.ArkUI'; ​ @Entry @Component struct HomePage { build() { Column({ space: 16 }) { // ✅ 方式1: pushUrl跳转(可返回) Button('查看详情') .onClick(() => { router.pushUrl({ url: 'pages/DetailPage' }); }) // ✅ 方式2: replaceUrl跳转(不可返回) Button('登录') .onClick(() => { router.replaceUrl({ url: 'pages/LoginPage' }); }) // ✅ 方式3: back返回 Button('返回') .onClick(() => { router.back(); }) } } }

2. 携带参数跳转

// 发送页面 @Component struct ItemList { onItemClick(item: Item): void { // ✅ 通过params传递参数 router.pushUrl({ url: 'pages/ItemDetailPage', params: { itemId: item.id, itemName: item.name, // ✅ 可以传递对象 itemData: item } }, router.RouterMode.Standard); } build() { List() { ForEach(this.items, (item: Item) => { ListItem() { Text(item.name) } .onClick(() => { this.onItemClick(item); }) }) } } } ​ // 接收页面 @Entry @Component struct ItemDetailPage { @State itemId: number = 0; @State itemName: string = ''; @State itemData: Item | null = null; aboutToAppear(): void { // ✅ 获取路由参数 const params = router.getParams() as Record<string, Object>; this.itemId = params.itemId as number; this.itemName = params.itemName as string; this.itemData = params.itemData as Item; console.info('接收参数:', this.itemId, this.itemName); } build() { Column() { Text(`ID: ${this.itemId}`) Text(`名称: ${this.itemName}`) } } }

3. 返回并传递结果

// 页面A: 打开页面B @Component struct PageA { @State selectedValue: string = ''; async openSelector(): Promise<void> { // ✅ 打开选择器页面 await router.pushUrl({ url: 'pages/SelectorPage' }); // ✅ 监听返回结果 router.getParams(); // 获取返回的数据 } aboutToAppear(): void { // ✅ 页面重新显示时获取结果 const params = router.getParams() as Record<string, Object>; if (params && params.selectedValue) { this.selectedValue = params.selectedValue as string; } } } ​ // 页面B: 选择后返回 @Component struct SelectorPage { onSelect(value: string): void { // ✅ 返回并携带数据 router.back({ url: 'pages/PageA', params: { selectedValue: value } }); } build() { List() { ListItem() { Text('选项1') } .onClick(() => { this.onSelect('选项1'); }) } } }

4. RouterMode 模式

// ✅ Standard模式(默认): 每次都创建新实例 router.pushUrl({ url: 'pages/DetailPage' }, router.RouterMode.Standard); ​ // ✅ Single模式: 复用已有实例 router.pushUrl({ url: 'pages/DetailPage', params: { id: 2 } }, router.RouterMode.Single); // 如果DetailPage已存在,会复用并更新参数

5. 路由拦截

/** * 路由守卫 */ export class RouterGuard { /** * 需要登录的页面 */ private static authPages: string[] = [ 'pages/ProfilePage', 'pages/SettingsPage', 'pages/OrderPage' ]; /** * 跳转前检查 */ static async navigate(url: string, params?: Object): Promise<void> { // ✅ 检查是否需要登录 if (this.authPages.includes(url)) { const isLoggedIn = await this.checkLogin(); if (!isLoggedIn) { // 未登录,跳转到登录页 router.pushUrl({ url: 'pages/LoginPage', params: { redirect: url, // 登录后跳转回来 redirectParams: params } }); return; } } // 已登录或不需要登录,正常跳转 router.pushUrl({ url, params }); } private static async checkLogin(): Promise<boolean> { const token = await AppSettings.getInstance().getUserToken(); return token !== null && token !== ''; } } ​ // 使用路由守卫 Button('我的订单') .onClick(() => { RouterGuard.navigate('pages/OrderPage'); })

6. 路由工具类

/** * 路由工具类 */ export class RouterUtils { /** * 跳转到详情页 */ static goToDetail(id: number): void { router.pushUrl({ url: 'pages/DetailPage', params: { id } }); } /** * 跳转到编辑页 */ static goToEdit(item: Item): void { router.pushUrl({ url: 'pages/EditPage', params: { item: JSON.stringify(item) } // ✅ 复杂对象转JSON }); } /** * 安全返回(检查是否有上一页) */ static safeBack(): void { const length = router.getLength(); if (length > 1) { router.back(); } else { // 没有上一页,跳转到首页 router.replaceUrl({ url: 'pages/Index' }); } } /** * 清空路由栈并跳转 */ static clearAndGo(url: string): void { router.clear(); // ✅ 清空路由栈 router.pushUrl({ url }); } /** * 获取参数(类型安全) */ static getParams<T>(): T | null { const params = router.getParams(); return params ? params as T : null; } } ​ // 使用 RouterUtils.goToDetail(123); RouterUtils.goToEdit(item); RouterUtils.safeBack();

关键 API

1. 路由跳转

方法说明是否可返回
pushUrl跳转到新页面
replaceUrl替换当前页面
back返回上一页-
clear清空路由栈-

2. 获取信息

// ✅ 获取路由参数 const params = router.getParams(); ​ // ✅ 获取路由栈长度 const length = router.getLength(); ​ // ✅ 获取当前路由状态 const state = router.getState(); console.info('当前页面:', state.path); console.info('路由名称:', state.name);

实战案例

案例 1: 列表-详情跳转

// 列表页 @Component struct ItemList { @State items: Item[] = []; build() { List() { ForEach(this.items, (item: Item) => { ListItem() { Row() { Text(item.name).fontSize(16); } .width('100%') .padding(16) } .onClick(() => { // ✅ 跳转到详情 router.pushUrl({ url: 'pages/ItemDetailPage', params: { itemId: item.id } }); }) }) } } } ​ // 详情页 @Entry @Component struct ItemDetailPage { @State item: Item | null = null; private itemId: number = 0; aboutToAppear(): void { // ✅ 获取参数 const params = router.getParams() as Record<string, Object>; this.itemId = params.itemId as number; // 加载详情 this.loadDetail(); } async loadDetail(): Promise<void> { this.item = await ItemDao.findById(this.itemId); } build() { Column() { // 返回按钮 Row() { Image($r('app.media.ic_back')) .width(24) .height(24) .onClick(() => { router.back(); }) } if (this.item) { Text(this.item.name).fontSize(20); } } } }

案例 2: 登录后跳转回原页面

// 登录页 @Entry @Component struct LoginPage { private redirectUrl: string = ''; aboutToAppear(): void { const params = router.getParams() as Record<string, Object>; this.redirectUrl = params?.redirect as string || 'pages/Index'; } async onLogin(): Promise<void> { // 登录成功 await login(); // ✅ 跳转回原页面 router.replaceUrl({ url: this.redirectUrl }); } }

最佳实践

1. 参数传递

// ✅ 推荐:传递ID,详情页加载数据 router.pushUrl({ url: 'pages/DetailPage', params: { id: 123 } // 只传ID }); // ❌ 不推荐:传递大对象 router.pushUrl({ url: 'pages/DetailPage', params: { item: largeObject } // 对象太大 });

2. 错误处理

try { await router.pushUrl({ url: 'pages/DetailPage' }); } catch (err) { console.error('跳转失败:', err); promptAction.showToast({ message: '页面不存在' }); }

3. 生命周期

@Entry @Component struct MyPage { // ✅ 页面首次创建 aboutToAppear(): void { const params = router.getParams(); // 初始化数据 } // ✅ 页面重新显示(从其他页面返回时) onPageShow(): void { // 刷新数据 this.refreshData(); } // ✅ 页面隐藏(跳转到其他页面时) onPageHide(): void { // 保存状态 } }

常见问题

Q1: 返回时如何传递数据?

使用 router.back()的 params 参数,在上一页通过 getParams 获取。

Q2: 如何清空路由栈?

使用 router.clear()清空,然后 pushUrl 跳转。

Q3: 如何实现 Tab 内导航?

使用 Navigation 组件,而不是 router。

总结

路由导航要点:

✅ pushUrl 跳转,back 返回 ✅ params 传递参数 ✅ RouterMode 控制实例 ✅ 路由守卫鉴权 ✅ 工具类封装常用跳转

参考资料

  • Router API
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/10 13:34:07

LobeChat能否实现对话导出为PDF或Markdown文件?

LobeChat能否实现对话导出为PDF或Markdown文件&#xff1f; 在今天&#xff0c;越来越多的开发者和知识工作者开始将AI聊天工具作为日常工作的核心助手。从撰写技术文档、整理学习笔记&#xff0c;到客户服务记录与团队协作问答&#xff0c;人们不再满足于“说完了就结束”的临…

作者头像 李华
网站建设 2026/6/10 15:12:03

(开题报告/毕业设计)基于springboot+vue智慧校园管理系统毕业项目源代码

springbootvue智慧校园管理系统 摘 要 传统校园管理系统往往由多个独立的子系统组成&#xff0c;如教务管理等&#xff0c;这些系统之间缺乏有效的数据共享和交互&#xff0c;且各类审批流程多为纸质化操作&#xff0c;流程复杂且耗时&#xff0c;降低了工作效率。随着信息技…

作者头像 李华
网站建设 2026/6/10 15:13:16

基于Seed-Coder-8B-Base的IDE插件设计思路与实现路径

基于Seed-Coder-8B-Base的IDE插件设计思路与实现路径 在现代软件开发中&#xff0c;开发者每天面对的是越来越复杂的项目结构、日益增长的代码量和不断提速的交付压力。尽管主流IDE已经提供了诸如语法高亮、自动补全、重构支持等基础辅助功能&#xff0c;但这些工具大多停留在“…

作者头像 李华
网站建设 2026/6/9 19:05:03

apk pure安卓应用风险高?转向桌面端AI工具更安全

从高风险APK到本地AI&#xff1a;为何Qwen3-32B正成为安全智能的新选择 在某金融科技公司的内部审计中&#xff0c;一次例行检查发现多名员工的手机上安装了一款名为“AI代码助手”的应用——它能快速解释复杂算法、生成Python脚本&#xff0c;甚至自动补全SQL查询。听起来很高…

作者头像 李华
网站建设 2026/6/10 4:43:34

火山引擎AI大模型对比:Qwen3-32B表现亮眼

火山引擎AI大模型对比&#xff1a;Qwen3-32B表现亮眼 在当前企业级AI应用的落地浪潮中&#xff0c;一个核心矛盾日益凸显&#xff1a;如何在保证模型智能水平的同时&#xff0c;控制部署成本与推理延迟&#xff1f;过去几年&#xff0c;千亿参数闭源模型凭借强大性能主导市场&a…

作者头像 李华
网站建设 2026/6/10 11:46:55

51单片机TM1804控制RGB灯闪烁的问题

今天在调RGB灯带时发现&#xff1a;颜色&#xff0c;数量&#xff0c;都能正常显示 但是就是每隔一会&#xff0c;某颗RGB灯都会闪一下&#xff0c; 正常&#xff1a;异常&#xff1a;&#xff08;某个灯闪烁&#xff09;最后发现是&#xff0c;是因为中断的影响 因为51单片机没…

作者头像 李华