第9篇|导航返回乱跳:用 NavPathStack 把返回路径收口

第9篇|导航返回乱跳:用 NavPathStack 把返回路径收口

摘要:实际项目里,导航最怕的不是“跳不过去”,而是“跳过去以后回不来”。首页、详情页、编辑页、登录页、通知入口混在一起后,如果每个按钮都自己决定 push、replace、back,返回路径很快就会失控。我更稳的做法是把 Navigation、NavPathStack、路由名和参数模型放到一条链路里,让页面进入和页面返回都能被解释。

我遇到过一个典型问题:从首页进入课程详情,再进评论页,返回正常;从通知入口进同一个详情页,再点评论页,返回却回到了首页;从搜索页进入时又变成直接退出当前业务。代码里每个入口看起来都没错,问题在于栈操作没有统一。

在这里插入图片描述

这篇文章解决四个具体问题:

  1. 为什么返回路径乱跳通常不是单个按钮的问题。
  2. 如何用路由名和参数模型收口入口差异。
  3. 什么时候该 push,什么时候该 replace,什么时候该 clear。
  4. 单实例页面重新带参数进入时,应该在哪里刷新。

在这里插入图片描述

在这里插入图片描述

先判断:你是在管理页面,还是在管理栈

页面跳转写到后期,经常会出现这种代码:

// pages/CourseCard.ets
Button('查看详情')
  .onClick(() => {
    router.pushUrl({
      url: 'pages/CourseDetailPage',
      params: { courseId: this.courseId }
    })
  })

如果项目已经在使用 Navigation,还继续在子页面里散落旧式路由调用,问题会越来越难定位。更稳的方向是把页面栈作为一个明确对象来管理,所有页面都通过同一个 NavPathStack 进入。

返回路径异常时,先不要改按钮文案,也不要先改页面生命周期。先问三个问题:

问题排查意义
当前页面是 push 进来的还是 replace 进来的决定 back 后是否还有上一页
这个入口是否应该保留来源页决定使用 push 还是 replace
目标页是否允许重复入栈决定是否使用单实例策略

这三个问题答清楚以后,返回问题就从“凭感觉修”变成了栈策略问题。

用路由名集中表达页面入口

我会先把页面路由名集中起来,避免入口里到处写字符串:

// navigation/AppRoutes.ets
export const AppRoutes = {
  home: 'Home',
  courseDetail: 'CourseDetail',
  courseComment: 'CourseComment',
  login: 'Login'
} as const

export interface CourseDetailParam {
  courseId: string
  from: 'home' | 'search' | 'notification'
}

这段代码的边界很清楚:路由名只在一个地方定义,参数也有稳定形状。后面搜索入口、通知入口、首页入口都传 CourseDetailParam,不会一处叫 id,另一处叫 course_id

Navigation 根节点只维护一份栈

NavigationNavPathStack 的关系要简单:一个 Navigation 只绑定一份栈,不要在多个组件里各自 new。

// pages/MainNavigationPage.ets
import { AppRoutes } from '../navigation/AppRoutes'

@Entry
@Component
struct MainNavigationPage {
  private pageStack: NavPathStack = new NavPathStack()

  aboutToAppear(): void {
    AppStorage.setOrCreate('app_nav_stack', this.pageStack)
  }

  build() {
    Navigation(this.pageStack) {
      HomePage()
    }
    .title('学习中心')
    .navDestination(this.routeBuilder)
  }

  @Builder
  routeBuilder(name: string) {
    if (name === AppRoutes.courseDetail) {
      CourseDetailPage()
    } else if (name === AppRoutes.courseComment) {
      CourseCommentPage()
    } else if (name === AppRoutes.login) {
      LoginPage()
    }
  }
}

这里的意图是让栈归宿主页面管理。子页面可以拿到这份栈,但不应该自己创建新栈。多份栈同时存在时,最常见的表现就是:某个页面看起来 push 了,返回却不按预期走。

页面发起跳转时只走封装方法

子页面不要直接写一堆栈操作。封装一个轻量导航服务,统一参数和操作语义:

// navigation/AppNavigator.ets
import { AppRoutes, CourseDetailParam } from './AppRoutes'

export class AppNavigator {
  static openCourseDetail(stack: NavPathStack, courseId: string, from: CourseDetailParam['from']): void {
    const param: CourseDetailParam = {
      courseId,
      from
    }
    stack.pushPathByName(AppRoutes.courseDetail, param)
  }

  static replaceToLogin(stack: NavPathStack): void {
    stack.replacePathByName(AppRoutes.login, null)
  }

  static backToHome(stack: NavPathStack): void {
    stack.clear()
  }
}

这段代码避免了入口各写各的。首页进入详情是 openCourseDetail,搜索进入详情也是 openCourseDetail,通知入口如果需要直接打开详情,也可以先进入宿主栈,再调用同一个方法。这样排查时只需要看导航服务。

push、replace、clear 不要混着用

很多返回乱跳来自一个原因:开发时把 pushreplaceclear 当成“都能跳过去”的工具。它们的语义完全不同。

操作适合场景返回结果
push从列表进详情、详情进评论返回上一页
replace登录成功后进入首页、启动页进入首页当前页被替换
clear退出登录、回到根入口清空业务栈

如果一个详情入口应该返回搜索页,就不要用 replace;如果登录成功后不希望返回登录页,就不要用 push。栈策略要跟业务路径一致。

单实例页面要在 onNewParam 里刷新

有些页面不应该重复入栈,比如消息详情、课程详情、播放器页。重复 push 会导致用户按很多次返回才能离开业务。可以使用单实例策略,让已有页面回到栈顶。

// navigation/AppNavigator.ets
static openCourseDetailSingle(stack: NavPathStack, courseId: string): void {
  const param: CourseDetailParam = {
    courseId,
    from: 'notification'
  }
  stack.pushPathByName(
    AppRoutes.courseDetail,
    param,
    undefined,
    true,
    LaunchMode.MOVE_TO_TOP_SINGLETON
  )
}

单实例有一个重点:页面第一次创建时可以在 onReady 读参数;已有页面被移动到栈顶时,要在 onNewParam 处理新参数。

// pages/CourseDetailPage.ets
@Component
struct CourseDetailPage {
  @State courseId: string = ''

  build() {
    NavDestination() {
      CourseDetailContent({ courseId: this.courseId })
    }
    .onReady((ctx: NavDestinationContext) => {
      this.applyParam(ctx.pathInfo.param)
    })
    .onNewParam((param: Object) => {
      this.applyParam(param)
    })
  }

  private applyParam(param: Object | undefined): void {
    const value = (param as CourseDetailParam)?.courseId
    if (typeof value === 'string' && value.length > 0) {
      this.courseId = value
    }
  }
}

如果只在 onReady 读参数,单实例页面再次回到栈顶时可能继续显示旧数据。这类问题看起来像页面缓存,其实是生命周期用错了。

返回结果不要靠全局临时变量传

编辑页返回列表时,很多人会用一个全局变量标记“刚刚保存过”。更稳的方式是用栈返回结果或统一刷新服务。

// pages/CourseCommentPage.ets
function finishComment(stack: NavPathStack): void {
  stack.pop()
}

如果上一页需要刷新,不要让评论页直接修改详情页内部状态。详情页在重新出现或收到返回结果后,从服务层重新读取评论摘要。这样页面之间不会互相知道对方的内部字段。

我会怎样复查导航栈

我会按入口逐条走:

  1. 首页进入详情,再返回首页。
  2. 搜索进入详情,再返回搜索页。
  3. 详情进入评论,评论返回详情,详情再返回来源页。
  4. 通知入口打开详情,确认不会叠出多个重复详情页。
  5. 退出登录后清空栈,确认返回键不会回到旧页面。

这套顺序能覆盖 push、replace、clear、单实例和跨入口参数。

常见问题和处理方式

现象常见原因处理方式
返回直接回首页入口用了 replace需要保留来源页时改成 push
返回要按很多次详情页重复入栈对重复业务页使用单实例策略
页面显示旧参数单实例只在 onReady 读参数在 onNewParam 里刷新
某个按钮点了没反应栈不是同一份实例根 Navigation 统一维护 NavPathStack

小结:导航稳定靠栈语义,不靠按钮临时修

导航返回乱跳,本质是栈策略没有被统一管理。把路由名、参数模型、栈实例和跳转方法集中起来,明确 push、replace、clear 的使用边界,再补上单实例参数刷新,返回路径就会变得可预测。页面越多,越不能让每个按钮自己决定怎么跳。

下载代码方式:https://pan.quark.cn/s/e2157c05e625 在信息技术领域中,数学问题的复杂求解在很大程度上依赖于数值计算,这在科学计算、工程分析以及数据分析等多个方面尤为重要。线性方程组的求解是数值计算中的一个核心且关键的问题,而雅克比迭代法作为一种有效策略,专门用于处理大规模稀疏线性方程组。这个资源提供了一段采用C++语言编写的雅克比迭代法源代码,配合附带的博客文章,能够帮助使用者深入掌握此方法的基本原理和实际应用。 雅克比迭代法,有时也被称作局部迭代方法,主要用于求解形式为 Ax = b 的线性方程组,其中矩阵A需满足对角占优的条件。对角占优的特性是指矩阵中每个对角线元素的绝对值要大于该行其他元素绝对值之和,这一性质确保了算法的收敛性能。该方法的实施基于矩阵A的雅克比矩阵J,其构成方式为 J = D - L - U,其中D、L和U分别代表矩阵A的对角线部分、下三角部分以及上三角部分。 迭代过程的数学表达式为:x(k+1) = J^-1 * b + (I - J^-1*A) * x(k),在此表达式中,x(k)表示第k次迭代的解向量,x(k+1)则是第k+1次迭代的解向量,I是单位矩阵。每次迭代都利用前一次得到的解来计算下一次的解,迭代会持续进行,直到解的精度达到预设标准或迭代次数达到最大限制。 在使用C++进行编程实现时,主要步骤包括: 1. 初始化阶段:设定初始解向量x(0),并明确迭代过程中的参数,例如最大迭代次数和容许的误差界限。 2. 构建雅克比矩阵:依据矩阵A的非对角元素来形成J矩阵。 3. 迭代计算:依照上述迭代公式计算新的解向量,并验证是否满足终止条件(即当前解与前一次解的差值小于设定的误差界限)。 4. 结果输出...
源码下载地址: https://pan.quark.cn/s/24e22475d2c3 采用SSM框架构建的果蔬生鲜超市平台,亦称为果蔬在线交易系统。其用户界面部分涵盖了:账号登录流程、新用户注册功能、购物车内容维护、订单状态监控、收货地点设置、商品检索服务、商品购买操作等。系统后台则由以下核心单元构成:用户账户维护、收货地址簿维护、商品分类维护、商品信息维护、货品出库单维护、订单状态跟踪、销售业绩统计、系统整体配置等。采用SSM框架构建的果蔬生鲜超市平台,亦称为果蔬在线交易系统。其用户界面部分涵盖了:账号登录流程、新用户注册功能、购物车内容维护、订单状态监控、收货地点设置、商品检索服务、商品购买操作等。系统后台则由以下核心单元构成:用户账户维护、收货地址簿维护、商品分类维护、商品信息维护、货品出库单维护、订单状态跟踪、销售业绩统计、系统整体配置等。采用SSM框架构建的果蔬生鲜超市平台,亦称为果蔬在线交易系统。其用户界面部分涵盖了:账号登录流程、新用户注册功能、购物车内容维护、订单状态监控、收货地点设置、商品检索服务、商品购买操作等。系统后台则由以下核心单元构成:用户账户维护、收货地址簿维护、商品分类维护、商品信息维护、货品出库单维护、订单状态跟踪、销售业绩统计、系统整体配置等。采用SSM框架构建的果蔬生鲜超市平台,亦称为果蔬在线交易系统。其用户界面部分涵盖了:账号登录流程、新用户注册功能、购物车内容维护、订单状态监控、收货地点设置、商品检索服务、商品购买操作等。系统后台则由以下核心单元构成:用户账户维护、收货地址簿维护、商品分类维护、商品信息维护、货品出库单维护、订单状态跟踪、销售业绩统计、系统整体配置等。
源码下载地址: https://pan.quark.cn/s/a4b39357ea24 在当前文档中,我们将详细研究如何运用Eclipse集成开发环境(IDE)的自定义CSS选项来调整其所有视窗的背景色调以及其他常用视窗的色调。Eclipse作为一个功能强大的开源开发平台,能够支持多种编程语言,包括Java、C++以及Python等。对于那些长时间运用Eclipse的开发专业人士而言,个性化界面色调能够显著提升工作舒适感和效率。让我们深入理解Eclipse的色彩配置机制。Eclipse依托于SWT(Standard Widget Toolkit)框架,允许用户通过调整主题和CSS样式来改变其视觉呈现。在默认设置下,Eclipse会采用系统级别的视窗色调,但用户可以通过覆盖特定的CSS文件来实现个性化定制,而无需触及操作系统本身的设置。 实施步骤1:定位Eclipse的CSS文件 Eclipse的CSS文件通常存储在以下路径位置: ``` <eclipse安装目录>\plugins\org.eclipse.platform_<version>\css ``` 此处,`<eclipse安装目录>`代表用户安装Eclipse的文件夹位置,`<version>`指代Eclipse的版本标识。 实施步骤2:对原始CSS文件进行备份 在进行任何修改之前,务必对原CSS文件进行备份操作,以便在出现问题时能够迅速恢复到原始状态。备份文件通常命名为`e4.css`和`e4_basestyle.css`。 实施步骤3:建立或编辑CSS文件 创建一个新的CSS文件(例如`custom_theme.css`),并插入以下内容以设定窗口背景色: ```css .e4-applicatio...
代码下载地址: https://pan.quark.cn/s/a4b39357ea24 根据所提供的文件资料可以判断,这是一份关于RS232转422/485转换器电路原理图的详尽设计文档。该文档全面地列出了电路中各个组件及其连接方法,对于掌握RS232与422/485之间的信号转换原理具有非常重要的参考意义。 ### RS232、RS422与RS485概述 在开始深入剖析电路原理图之前,有必要先对RS232、RS422与RS485这三种通信协议的基本概念进行简要介绍。 #### RS232 RS232是一种应用于串行数据通信的接口规范,主要适用于计算机与调制解调器或其他外部设备之间的数据交互。该接口标准支持点对点的通信模式,通信距离通常不超过15米,并且较为容易受到外界干扰的影响。 #### RS422 RS422是一种经过改进的串行通信标准,其核心特点在于采用差分信号进行传输,支持多点的通信模式,即一个发送端能够同时向多个接收端传输数据,通信距离最远可达1200米,并且具有较强的抗干扰性能。 #### RS485 RS485是建立在RS422基础之上的进一步发展,同样运用差分信号进行传输,其最突出的特点在于支持半双工通信模式,即在同一时刻只能进行发送或接收操作,但发送端和接收端的位置可以互换,非常适合于长距离、多设备之间的数据传输,通信距离同样可以达到1200米,并且能够支持多达32个设备接入。 ### RS232转422/485转换器电路解析 在电路原理图中,可以观察到采用了MAX490CPA和MAX485CPA芯片作为RS232与RS422/485之间的信号转换装置,同时使用了MAX233ACPP芯片作为RS232电平转换装置。 #### MAX490CP...
内容概要:本文档系统汇集了“计及电动汽车充电站接入的配电网承载能力评估与优化”的Matlab代码实现资源,覆盖无功优化、多时间尺度调度、N-1/N-k故障分析、电动汽车V2G技术、微电网协调调度、电氢耦合系统、风光储联合系统等多个电力系统前沿研究方向。资源以Matlab/Simulink为核心工具,辅以Python,提供大量可复现的科研代码实例,涵盖从建模、优化算法(如NSGA-II、DDPG、MPC、PSO等)到仿真验证的全流程。同时拓展至机器学习、深度学习、路径规划、信号处理、无人机控制、综合能源系统优化等多个交叉领域,配套网盘资料与公众号支持,助力科研人员高效开展创新研究与高水平论文复现。; 适合人群:具备电力系统基础知识和Matlab编程能力,从事电气工程、能源互联网、智能电网、综合能源系统等方向研究的研究生、科研人员及工程技术人员。; 使用场景及目标:①研究电动汽车大规模接入对配电网安全性、稳定性及承载力的影响;②构建含V2G的无功优化与电压协同控制模型;③实现多时间尺度下微电网与配电网的协调调度;④复现N-1/N-k故障下的安全约束调度、鲁棒恢复等复杂优化模型;⑤开展综合能源系统、电氢氨耦合系统等新兴领域的仿真与优化研究; 阅读建议:建议结合提供的网盘资源与公众号内容体系化学习,优先掌握核心案例的算法架构与建模逻辑,按研究主题分类深入,并注重将理论方法与代码实践紧密结合,提升科研效率与创新能力。
已经博主授权,源码转载自 https://pan.quark.cn/s/baa6fed0ef20 **SHT20温湿度传感器简介** SHT20是由瑞士Sensirion公司制造的精密数字型温湿度检测设备,在环境检测、智能家居、农业、医疗装置及工业自动化等多个领域得到普遍运用。该传感器凭借其高精确度、低能耗以及稳定的运作表现,赢得了业界的广泛认可。 **传感器工作机制** SHT20的核心技术在于电容式湿度检测与NTC热敏电阻的温度测量。湿度检测部分通过感知水分子对传感器表面硅聚合物膜层的作用来判定相对湿度(RH),而温度检测部分则借助热敏电阻的电阻值随温度变化的特性来计量环境温度。 **核心优势** 1. **高精确度**:SHT20能够提供±2%RH的湿度检测准确度和±0.3°C的温度检测准确度,在同类型产品中表现卓越。 2. **低能耗**:在设计中注重节能,适合用于电池供电或能量收集系统,工作电流可小至1.1μA。 3. **迅速响应**:响应速度快,能迅速适应环境变化,对于实时监测具有关键意义。 4. **I²C接口**:采用标准的I²C数字接口,便于融入各种微控制器系统中。 5. **全范围测量**:支持0% to 100%RH的湿度和-40°C to +125°C的温度测量范围。 6. **紧凑封装**:小型化封装设计,节省空间,方便安装在有限空间内。 **使用场景** 1. **智能家居**:用于智能空调、加湿器和除湿机,实现室内环境自动调节以提升舒适度。 2. **农业管理**:温室环境监控,协助控制作物生长的最佳条件。 3. **医疗装置**:呼吸机、冷藏设备等,保障药品和样本的储存条件。 4. **环境监测**:空气质量监测站,气象站,提供精确的气象...
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值