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

鸿蒙应用开发UI基础第三十节:循环渲染核心ForEach 实战与性能优化

【学习目标】掌握 ForEach 循环渲染核心语法、键值生成规则与组件创建逻辑理解 ForEach 首次渲染与非首次渲染的差异避免渲染异常规避 ForEach 常见错误键值重复、性能损耗、数据不渲染掌握 ForEach 性能优化技巧实现高效循环渲染。一、本节工程目录结构ForEachDemo/ ├── entry/ │ └── src/main/ │ ├── ets/ │ │ ├── components/ │ │ │ ├── CustomListItem.ets # 自定义列表组件 │ │ ├── pages/ │ │ │ ├── Index.ets # 首页功能导航入口 │ │ │ ├── BasicRenderDemo.ets # 基础渲染与键值示例 │ │ │ └── SkeletonListDemo.ets # 静态列表/骨架屏示例 │ └── resources/ │ └── base/media/ # 图片资源 └── build-profile.json5二、ForEach 核心认知2.1 什么是 ForEachForEach 是鸿蒙 ArkTS 提供的循环渲染接口基于数组数据源批量生成组件需与容器组件配合使用核心价值是简化重复组件的开发提升代码复用性。2.2 核心语法ForEach(dataSource:ArrayT,// 数据源数组类型itemGenerator:(item:T,index?:number)void,// 组件生成函数遍历数组项创建组件keyGenerator?:(item:T,index?:number)string// 键值生成函数可选默认按 indexJSON.stringify(item) 生成)2.3 键值生成规则键值Key是 ForEach 识别组件唯一性的核心用于判断组件是否需要创建/复用/销毁若自定义keyGenerator需返回唯一且持久的字符串如对象的唯一 ID若未自定义框架默认生成规则为index __ JSON.stringify(item)键值重复后果框架仅创建第一个重复键值对应的组件后续重复项不渲染键值变化后果框架视为组件替换销毁旧组件并创建新组件。三、ForEach 渲染机制详解3.1 首次渲染遍历数据源为每个数组项生成唯一键值根据键值创建对应组件并渲染流程遍历数组每一项调用keyGenerator生成键值若键值未存在创建组件并记录键值若键值已存在跳过组件创建避免重复渲染。自定义列表项// 子组件自定义列表项Componentexportstruct CustomListItem{Prop content:string;aboutToAppear():void{console.log(CustomListItem - 组件即将出现 ----${this.content})}aboutToDisappear():void{console.log(CustomListItem - 组件即将消失 ----${this.content})}build(){Text(this.content).fontSize(18).padding(15).backgroundColor(#F7F8FA).borderRadius(8).width(100%)}}ForEach基础渲染示例import{CustomListItem}from../components/CustomListItem;Entry Component struct BasicRenderDemo{State fruitList:string[][苹果,香蕉,橙子,葡萄];build(){Column({space:12}){Text(ForEach核心细节).fontSize(20).fontWeight(FontWeight.Bold).margin({bottom:10})// ForEach 首次渲染无重复项ForEach(this.fruitList,(item:string)CustomListItem({content:item}),// 生成组件(item:string)item// 键值为数组项本身确保唯一)}.padding(20).width(100%).height(100%).backgroundColor($r(sys.color.point_color_checked))}}首次渲染—无重复项修改数据源// 数据源包含重复项State fruitList:string[][苹果,香蕉,香蕉,葡萄];首次渲染—有重复项仅渲染「苹果、香蕉、葡萄」三项重复的「香蕉」因键值重复被跳过。3.2 非首次渲染数据源变化后当数据源发生变化增删改查时框架触发重新渲染流程遍历新数据源生成新键值列表对比新旧键值列表新键值不存在创建新组件新键值已存在复用旧组件不重新创建旧键值不存在于新列表销毁对应组件。修改示例Entry Component struct BasicRenderDemo{State fruitList:string[][苹果,香蕉,香蕉,葡萄];// 修改第三项值updateThirdItem(){this.fruitList[2]芒果;}// 插入第一行insertFirstItem(){this.fruitList.splice(0,0,车厘子);}build(){Column({space:12}){Text(ForEach核心细节).fontSize(20).fontWeight(FontWeight.Bold).margin({bottom:10})Button(修改第三项为芒果).width(100%).backgroundColor($r(sys.color.brand)).fontColor(Color.White).onClick(()this.updateThirdItem())Button(新增插入第一行车厘子).width(100%).backgroundColor($r(sys.color.brand)).fontColor(Color.White).onClick(()this.insertFirstItem())ForEach(this.fruitList,(item:string)CustomListItem({content:item}),// 生成组件(item:string)item// 键值为数组项本身确保唯一)}.padding(20).width(100%).height(100%).backgroundColor($r(sys.color.point_color_checked))}}点击修改第三项为芒果按钮后原本重复项香蕉只创建了一个修改后键值改变无重复项创建了芒果其他组件复用。CustomListItem - 组件即将出现 ----苹果 CustomListItem - 组件即将出现 ----香蕉 CustomListItem - 组件即将出现 ----芒果 CustomListItem - 组件即将出现 ----葡萄点击插入第一行车厘子按钮后新创建了车厘子其他组件复用。CustomListItem - 组件即将出现 ---- 车厘子修改键值使用系统默认index __ JSON.stringify(item)当使用 index 进行新增、删除、修改数据源后键值会变动造成组件重复销毁与创建产生性能损耗。ForEach(this.fruitList,(item:string)CustomListItem({content:item}),// 生成组件// (item:string) item // 键值为数组项本身确保唯一)首次运行日志输出组件全部显示CustomListItem - 组件即将出现 ---- 苹果 CustomListItem - 组件即将出现 ---- 香蕉 CustomListItem - 组件即将出现 ---- 香蕉 CustomListItem - 组件即将出现 ---- 葡萄点击插入第一行车厘子按钮后数组存储[车厘子,苹果,香蕉,香蕉,葡萄]键值改变为车厘子_0 苹果_1 香蕉_2 香蕉_3 葡萄_4而原键值为苹果_0 香蕉_1 香蕉_2 葡萄_3。此时香蕉_2无变动其他组件键值已变化日志输出如下CustomListItem - 组件即将消失 ---- 苹果 CustomListItem - 组件即将消失 ---- 葡萄 CustomListItem - 组件即将消失 ---- 香蕉 CustomListItem - 组件即将出现 ---- 车厘子 CustomListItem - 组件即将出现 ---- 苹果 CustomListItem - 组件即将出现 ---- 香蕉 CustomListItem - 组件即将出现 ---- 葡萄示例 List ForEachimport{CustomListItem}from../components/CustomListItem;Entry Component struct ListRenderDemo{State fruitList:string[][];aboutToAppear():void{for(letindex0;index100;index){this.fruitList.push(组件${index})}}// 修改第三项值updateThirdItem(){this.fruitList[2]芒果;}// 插入第一行insertFirstItem(){this.fruitList.splice(0,0,车厘子);console.log(JSON.stringify(this.fruitList))}build(){Column({space:12}){Text(List ForEach核心细节).fontSize(20).fontWeight(FontWeight.Bold).margin({bottom:10})Button(修改第三项为芒果).width(100%).backgroundColor($r(sys.color.brand)).fontColor(Color.White).onClick(()this.updateThirdItem())Button(新增插入第一行车厘子).width(100%).backgroundColor($r(sys.color.brand)).fontColor(Color.White).onClick(()this.insertFirstItem())this.scrollerBuilder()// this.listBuilder()}.padding(20).width(100%).height(100%).backgroundColor($r(sys.color.point_color_checked))}LocalBuilderscrollerBuilder(){Scroll(){Column(){ForEach(this.fruitList,(item:string){CustomListItem({content:item})// 生成组件},(item:string)item)}}.width(100%).layoutWeight(1)}LocalBuilderlistBuilder(){List(){ForEach(this.fruitList,(item:string){ListItem(){CustomListItem({content:item})// 生成组件}},(item:string)item)}.width(100%).layoutWeight(1)}}scrollerBuilder 首次运行CustomListItem - 组件即将出现 ----组件0 ... 连续输出 CustomListItem - 组件即将出现 ----组件99检查视图组件从0创建到99总计100条可视区域仅10条不可视区域也同样创建并挂载到组件树上。如果是1000条数据ForEach依然会全部创建渲染性能极低内存消耗极大。使用ListForEachlistBuilder// this.scrollerBuilder()this.listBuilder()再次运行查看日志配合List组件后循环渲染性能有所提升。从0创建到23并未全部创建。页面可视区域显示11条0-10不可见区域预创建13条。检查UI视图我们发现ListForEach循环渲染组件性能大幅提升。虽然首次运行创建了23个组件可视区域显示10条但不可见区域的13条并未直接挂载到组件树上。点击修改第三项为芒果销毁原来的组件2新创建芒果其他组件复用。CustomListItem - 组件即将消失 ---- 组件2 CustomListItem - 组件即将出现 ----芒果滑动列表查看日志向上滑动屏幕直到组件10消失组件11成为可视区域第一行。日志开始继续输出从第24条至34条停止。由此可见List会提前预创建组件在即将显示时挂载到组件树上。CustomListItem - 组件即将出现 ---- 组件24 CustomListItem - 组件即将出现 ---- 组件25 ..... CustomListItem - 组件即将出现 ---- 组件33 CustomListItem - 组件即将出现 ---- 组件34通过以上演示List虽然提升了加载性能但使用ForEach修改State状态变量时仍会销毁并重新创建组件。使用index作为键值数据源变化后键值会重新排序销毁State之后的所有组件并重新创建。因此ForEach仅适合静态少量列表、数据源不变的场景。注意使用JSON.stringify(item)作为键值时数据量不宜过大内存会飙升若数据中包含bigint类型JSON.stringify会直接抛出错误导致程序无法正常运行。四、ForEach适用场景适用于静态无变动的数据源列表、数据量较小通常少于100条。例如骨架屏、固定选项列表、设置页等场景。示例骨架屏静态渲染Entry Component struct SkeletonListDemo{// 静态数据源仅用于控制骨架屏数量State skeletonCount:number[][1,2,3,4,5];State translateX:string-100%;// 动画偏移量State duration:number1500;// 动画速度aboutToAppear():void{// 页面渲染后自动开始流光动画this.startShineAnimation();}// 流光动画无限循环startShineAnimation(){// 延迟100ms等待组件渲染完成否则动画会因组件未挂载而不生效setTimeout((){this.translateX-100%;// 设置起始位置this.getUIContext()?.animateTo({duration:this.duration,iterations:-1,// 无限循环curve:Curve.Linear},(){// 动画事件回调 让渐变组件移动100%this.translateX100%;});},100)}// 流光遮罩层LocalBuildershineOverlay(){Row().width(100%).height(100%).translate({x:this.translateX}).linearGradient({// 线性渐变angle:90,colors:[[rgba(255,255,255,0),0],[rgba(255,255,255,0.8),0.5],[rgba(255,255,255,0),1]]})}LocalBuilderskeletonItem(){Column({space:8}){Row().width(60).height(60).backgroundColor(#ECECEC).borderRadius(8)Row().width(80%).height(20).backgroundColor(#ECECEC).borderRadius(4)Row().width(70%).height(20).backgroundColor(#ECECEC).borderRadius(4)}.padding(10).backgroundColor(#F7F8FA).borderRadius(8).width(100%).alignItems(HorizontalAlign.Start).overlay(this.shineOverlay()).clip(true);// 必须加否则流光会超出圆角}build(){Column({space:12}){ForEach(this.skeletonCount,(item:number){this.skeletonItem()},(item:number)item.toString())}.padding(20).width(100%).height(100%).backgroundColor(#F1F3F5)}}运行效果五、ForEach 性能优化技巧键值优化优先使用对象唯一 ID 作为键值避免使用 index 或动态变化的字段组件复用子组件尽量轻量化避免在itemGenerator中创建复杂组件如嵌套多层布局数据源优化大数据量场景千条以上建议使用LazyForEach懒加载而非 ForEach避免混用List/Grid 容器中不要同时使用 ForEach 和 LazyForEach。六、内容总结核心逻辑ForEach 基于键值判断组件创建/复用/销毁键值必须唯一渲染机制首次渲染创建所有组件非首次渲染仅处理键值变化的组件场景选型静态列表基本类型数组 数组项作为键值动态列表对象数组 唯一 ID 作为键值避坑核心不使用 index 作为键值不省略keyGenerator性能优化轻量化子组件、优化键值、大数据量用 LazyForEach、长列表用 List 容器。七、代码仓库工程名称ForEachDemo仓库地址https://gitee.com/HarmonyOS-UI-Basics/harmony-os-ui-basics.git八、下节预告下一节我们将深入学习鸿蒙应用开发中的长列表性能优化核心 —— LazyForEach 懒加载实战掌握千条级数据列表的流畅滚动实现方案。

相关文章:

鸿蒙应用开发UI基础第三十节:循环渲染核心ForEach 实战与性能优化

【学习目标】 掌握 ForEach 循环渲染核心语法、键值生成规则与组件创建逻辑;理解 ForEach 首次渲染与非首次渲染的差异,避免渲染异常;规避 ForEach 常见错误(键值重复、性能损耗、数据不渲染);掌握 ForEach…...

Stm32f103c8t6(proteus仿真)进阶——PWM精准调控LED渐变效果

1. PWM基础与LED渐变原理 PWM(脉冲宽度调制)就像用开关快速控制灯泡的亮灭。想象你用手指快速按动电灯开关,按下的时间越长,灯泡看起来就越亮。这就是PWM控制LED亮度的基本原理。在STM32中,我们通过定时器产生这种快速…...

马斯克投200亿建芯片厂+小鹏成立Robotaxi部+20亿具身智能基金落地

1. 歌华有线与张江集团联合设立20亿具身智能机器人基金牛喀网获悉,海顺新具身智能机器人私募投资基金合伙企业(有限合伙)正式成立,出资额达20亿元,经营范围涵盖股权投资、资产管理等。该基金由歌华有线、上海张江集团全资子公司上海张江科技创…...

虚拟机跑Gazebo黑屏?3个实测有效的修复方案(含国内镜像加速)

虚拟机运行Gazebo黑屏问题全解析:从镜像加速到性能优化的实战指南 当你在虚拟化环境中搭建机器人仿真平台时,Gazebo的黑屏问题可能成为阻碍开发进度的"拦路虎"。特别是在国内网络环境下,模型加载缓慢、图形渲染异常等问题频发。本文…...

CAD 几何内核底层数学分类

以下对 OCCT (OpenCASCADE)、Parasolid 和 ACIS 三大几何内核所依赖的底层数学进行系统分类。一、基础数学 1.1 线性代数内容应用场景向量运算(点积、叉积)法线计算、方向判定、共面/共线检测矩阵运算(33, 44 齐次矩阵)几何变换&a…...

智能诊断技术在电机故障预测中的应用与挑战

1. 智能诊断技术如何革新电机故障预测 想象一下,你正在管理一个大型工厂的生产线,突然一台关键电机毫无征兆地停机了。这不仅意味着昂贵的维修费用,更可能导致整条生产线瘫痪,造成每小时数十万元的经济损失。这正是传统电机维护方…...

DolphinScheduler vs Airflow:跨工作流依赖实现机制深度对比(附性能测试数据)

DolphinScheduler与Airflow跨工作流依赖机制全景对比:架构设计与性能实战 1. 调度系统演进与核心挑战 在现代数据工程实践中,工作流调度系统已成为数据处理管道的核心中枢。随着数据规模的指数级增长和业务复杂度的提升,传统简单的定时任务调…...

RISC-V开发实战:手把手教你用wfi指令优化嵌入式系统功耗(附代码示例)

RISC-V开发实战:用wfi指令实现嵌入式系统低功耗优化的完整指南 在嵌入式系统开发中,功耗优化一直是工程师们面临的永恒挑战。想象一下,你的设备需要长时间等待一个外部事件——可能是用户按下按钮,或是传感器检测到特定阈值——而…...

从 “瞎猜” 到 “精准”:一文搞懂 LLM reAct 框架的底层逻辑与实战

核心概念:打破“幻觉”的循环想象一下,你正在主导设计一个复杂的业务系统,比如一个需要处理海量高校数据的智能助手。如果用户问大模型:“找出计算机系上个月登记的、采用率最高的三本教材,并核对它们在最新系统里的定…...

Qwen3-32B-Chat私有化部署案例:金融研报摘要生成服务API封装

Qwen3-32B-Chat私有化部署案例:金融研报摘要生成服务API封装 1. 项目背景与价值 在金融行业,每天都会产生大量研究报告,分析师需要花费大量时间阅读和提炼关键信息。传统的人工摘要方式效率低下,且难以保证一致性。Qwen3-32B作为…...

Python海龟绘图动画教程:如何用turtle模块制作颜色变化效果

Python海龟绘图动画实战:打造流畅颜色渐变效果 1. 初识turtle模块的动画潜力 Python的turtle模块远不止是一个简单的绘图工具,它内置的动画机制能让图形"活"起来。与其他静态绘图库不同,turtle的独特之处在于它能实时展示绘图过程&…...

Issac Sim+VScode高效开发:5个提升调试效率的隐藏技巧(含RL案例)

Issac SimVScode高效开发:5个提升调试效率的隐藏技巧(含RL案例) 在机器人仿真与强化学习开发领域,NVIDIA Issac Sim凭借其强大的物理引擎和Omniverse生态已成为行业标杆工具。但许多开发者可能不知道,当它与VScode深度…...

Synopsys VCS最新版在Vivado 2023中的5个高效仿真技巧

Synopsys VCS 2024与Vivado 2023协同仿真的5个高阶技巧 在FPGA设计验证领域,仿真效率直接决定了项目迭代速度。当Synopsys VCS 2024遇上Xilinx Vivado 2023,新特性的碰撞会产生怎样的火花?本文将揭示如何通过深度调优让仿真速度提升300%的实战…...

Pixel Dimension Fissioner惊艳对比:温度值0.3 vs 1.2的创意发散可视化

Pixel Dimension Fissioner惊艳对比:温度值0.3 vs 1.2的创意发散可视化 1. 工具概览 像素语言维度裂变器(Pixel Dimension Fissioner)是一款基于MT5-Zero-Shot-Augment核心引擎构建的文本改写与增强工具。它将传统AI工具的工业感转化为16-bit像素冒险风格&#xf…...

Vue项目PC端自适应终极方案:px2rem-loader+postcss-px2rem保姆级教程

Vue项目PC端自适应终极方案:px2rem-loaderpostcss-px2rem保姆级教程 在当今多设备、多分辨率的互联网环境下,前端开发者面临的最大挑战之一就是如何确保网站在不同尺寸的屏幕上都能完美呈现。特别是对于企业级Vue项目,PC端的自适应需求往往比…...

Google支付OR-BAIH-01错误代码全解析:从原因到修复的完整指南

Google支付OR-BAIH-01错误代码全解析:从原因到修复的完整指南 当你在Google Play商店尝试购买应用、游戏或订阅服务时,突然弹出一条错误提示:"發生未預期的錯誤。請改用其他付款方式繼續操作,或是與我們聯絡。瞭解詳情[OR-B…...

Qwen3-ASR-0.6B开源大模型落地指南:政务热线录音→工单分类→情感倾向分析

Qwen3-ASR-0.6B开源大模型落地指南:政务热线录音→工单分类→情感倾向分析 1. 项目简介与核心价值 Qwen3-ASR-0.6B是阿里云通义千问团队推出的轻量级语音识别模型,专门为本地化部署场景设计。这个6亿参数的模型在保持较高识别精度的同时,大…...

Ubuntu 22.04下用Docker搞定YOLOv5/v7训练环境:从驱动安装到镜像构建全流程

Ubuntu 22.04下用Docker搞定YOLOv5/v7训练环境:从驱动安装到镜像构建全流程 在计算机视觉领域,YOLO系列算法因其出色的实时检测性能而广受欢迎。然而,搭建一个稳定、高效的YOLO训练环境往往让开发者头疼不已——不同版本的CUDA、PyTorch和系…...

OpenClaw配置优化指南:提升GLM-4.7-Flash任务执行稳定性

OpenClaw配置优化指南:提升GLM-4.7-Flash任务执行稳定性 1. 问题背景与挑战 去年冬天,当我第一次尝试用OpenClaw对接GLM-4.7-Flash模型完成自动化文档整理任务时,遇到了令人头疼的问题——任务执行到一半突然中断,控制台只留下一…...

避坑指南:PaddleOCR多语言模型部署常见问题排查(韩文/日文实例)

避坑指南:PaddleOCR多语言模型部署常见问题排查(韩文/日文实例) 当你在国际化项目中部署PaddleOCR处理韩文或日文文档时,是否遇到过识别结果全是乱码?或者明明安装了字体却显示为方框?这些问题往往源于多语…...

手把手教你调试Linux下的lt8619c.c驱动:从设备树解析到V4L2控件初始化

手把手教你调试Linux下的lt8619c.c驱动:从设备树解析到V4L2控件初始化 在嵌入式Linux开发中,显示接口驱动调试往往是项目推进的关键环节。LT8619C作为一款高性能HDMI接收芯片,其驱动开发涉及设备树配置、V4L2框架集成、中断处理等多个技术要点…...

JS 原型链,一篇文章让你彻底记住(忘都忘不掉)

👉 JS 原型链,一篇文章让你彻底记住(忘都忘不掉)1. 问题背景(真实场景) 你一定遇到过这些情况: function Person(name) {this.name name; }Person.prototype.sayHi function () {console.log(…...

LSM9DS1九轴IMU驱动开发与硬件协同设计指南

1. LSM9DS1九轴惯性测量单元技术解析与嵌入式驱动开发实践LSM9DS1是意法半导体(STMicroelectronics)推出的一款高集成度、低功耗的九轴惯性测量单元(IMU),集成了三轴加速度计、三轴陀螺仪和三轴磁力计于单一封装内。该…...

基于CPO - TCN - BiGRU - Attention的多变量时间序列预测

Matlab完整源码和数据 1.基于CPO-TCN-BiGRU-Attention冠豪猪算法优化时间卷积双向门控循环单元融合注意力机制多变量时间序列预测,要求Matlab2023版以上; 2.输入多个特征,输出单个变量,考虑历史特征的影响,多变量时间序…...

避坑指南:Android无障碍服务中模拟Enter键的5个常见错误及解决方案

Android无障碍服务中模拟Enter键的五大实战陷阱与破解之道 在移动应用开发领域,无障碍服务(AccessibilityService)为开发者提供了强大的系统级交互能力,其中模拟键盘Enter键操作是实现自动化流程的关键技术点。然而,从API版本差异到节点查找策…...

用Qt Designer玩转UI设计:拖拽实现计算器界面实战

用Qt Designer玩转UI设计:拖拽实现计算器界面实战 在当今快速迭代的软件开发领域,GUI(图形用户界面)设计效率直接关系到产品的市场竞争力。作为跨平台C框架Qt的核心组件,Qt Designer以其"所见即所得"的设计理…...

计算机毕业设计:基于 Python双协同过滤的图书推荐系统 Django 协同过滤算法 可视化 数据分析 爬虫 大数据 机器学习(建议收藏)✅

博主介绍:✌全网粉丝50W,前互联网大厂软件研发、集结硕博英豪成立软件开发工作室,专注于计算机相关专业项目实战6年之久,累计开发项目作品上万套。凭借丰富的经验与专业实力,已帮助成千上万的学生顺利毕业,…...

LSV实战:5分钟搞定倾斜摄影+BIM场景搭建(附模型快速复制技巧)

LSV高效场景搭建:倾斜摄影与BIM模型融合实战指南 在数字化设计与城市规划领域,将倾斜摄影模型与BIM人工模型结合已成为行业标配工作流。这种融合技术能快速构建高精度三维场景,大幅提升规划展示效果与方案沟通效率。对于经常需要处理大型场景…...

yz-女生-角色扮演-造相Z-Turbo开源模型社区贡献指南

yz-女生-角色扮演-造相Z-Turbo开源模型社区贡献指南 1. 引言 大家好,今天我们来聊聊如何为yz-女生-角色扮演-造相Z-Turbo这个开源模型项目做贡献。如果你对这个模型感兴趣,想要参与改进或者添加新功能,这篇指南会告诉你具体该怎么做。 这个…...

如何绕过Physics2DPlugin3的试用版跳转限制(附修改后的JS文件)

Physics2DPlugin3本地化开发解决方案与GSAP整合实践 1. 理解Physics2DPlugin3的试用机制 Physics2DPlugin3作为GSAP生态中的重要物理动画插件,其试用版设计了一套域名检测机制。当检测到运行环境不在授权域名列表时,会自动触发跳转逻辑。这套机制主要通过…...