【鸿蒙HarmonyOS Next实战开发】实现组件动态创建和卸载-优化性能

ops/2025/2/21 4:35:05/
一、简介

为了解决页面和组件加载缓慢的问题,ArkUI框架引入了动态操作功能,支持组件的预创建,并允许应用在运行时根据实际需求动态加载和渲染组件。

这些动态操作包括动态创建组件(即动态添加组件)动态卸载组件(即动态删除组件)

动态创建组件是指在build生命周期之外提前创建组件,这种方式不仅能节省组件创建时间,提升用户体验,还能将独立逻辑封装起来,助力应用的模块化开发。

而动态卸载组件则是对动态创建的组件进行卸载和删除操作。

二、组件预创建原理

在传统的声明式开发范式中,组件仅在build阶段被创建,开发者无法在其他生命周期阶段进行组件创建,这往往会导致页面加载速度较慢。

然而,ArkUI框架提供的UI动态操作支持组件的预创建,允许开发者在非build生命周期阶段提前创建组件。这些预创建的组件可以在页面加载时直接使用,从而大幅提升页面的响应速度

如图1所示,利用组件预创建机制,可以在动画执行过程中的空闲时间进行组件预创建和属性设置。当动画结束后,再进行属性和布局的更新,从而节省组件创建时间,加快页面渲染速度

图1组件预创建原理图 

三、FrameNode自定义节点在动态布局场景下的优势
1.减少自定义组件创建开销


在声明式开发范式中,使用ArkUI的自定义组件对节点树中的每个节点进行定义时,往往会遇到节点创建效率低下的问题。

这是因为在ArkTS引擎中,每个节点都需要分配内存空间来存储应用程序的自定义组件和状态变量,并且在节点创建过程中还需执行组件ID、组件闭包以及状态变量之间的依赖关系收集等操作。相比之下,使用ArkUI的FrameNode可以避免创建自定义组件对象和状态变量对象,无需进行依赖收集,从而显著提升组件创建速度。

2.组件更新更快


在动态布局框架的更新场景中,通常存在一个由树形数据结构ViewModelA创建的UI组件树TreeA。当需要使用新的数据结构ViewModelB来更新TreeA时,尽管声明式开发范式可以实现数据驱动的自动更新,但这一过程中伴随着大量的diff操作。

对于ArkTS引擎而言,在对一个复杂组件树(深度超过30层,包含100至200个组件)执行diff算法时,几乎无法在120Hz的刷新率下保持满帧运行。然而,使用ArkUI的FrameNode扩展,框架能够自主掌控更新流程,实现高效的按需剪枝。特别是对于那些仅服务于少数特定业务的动态布局框架,利用这一扩展可以实现快速的更新操作。

3.直接操作组件树


在声明式开发范式中,组件树结构更新操作较为困难。例如,将组件树中的一个子树从当前子节点完整移到另一个子节点时,声明式开发范式无法直接调整组件实例的结构关系,只能通过重新渲染整棵组件树来实现。

而使用ArkUI的FrameNode扩展,则可以通过操作FrameNode轻松操控该子树,将其移植到另一个节点,从而实现局部渲染刷新,性能更优。

四、组件动态添加、更新和删除
1、动态添加组件


动态添加组件的过程包括以下步骤:

  1. 创建自定义节点。

  2. 实现NodeController,用于管理自定义节点的创建、显示、更新等操作,并负责将自定义节点挂载到NodeContainer上。

  3. 实现NodeController的makeNode方法,该方法会在NodeController实例绑定NodeContainer时被回调,并将返回的节点挂载至NodeContainer。

  4. 使用NodeContainer显示自定义节点。

2、创建自定义节点


首先,准备好需要挂载的节点,代码如下所示:

import { BuilderNode, FrameNode, NodeController } from '@kit.ArkUI';class Params {text: string = 'Hello World';constructor(text: string) {this.text = text;}
}@Builder
function buildText(params: Params) {Column() {Text(params.text).fontSize(50).fontWeight(FontWeight.Bold).margin({bottom: 36})}
}
3、实现NodeController


NodeController是一个抽象类,需要继承并实现它,代码如下所示:

class TextNodeController extends NodeController {private textNode: BuilderNode<[Params]> | null = null;private message: string = '';constructor(message: string) {super();this.message = message;}makeNode(context: UIContext): FrameNode | null {return null;}
}
4、实现NodeController的makeNode方法


首先,使用构造函数创建BuilderNode实例。创建BuilderNode对象时,必须传入对应的UIContext对象。如果BuilderNode作为RenderNode的子节点存在,则需要设置RenderOptions的selfIdealSize属性。然后,使用BuilderNode的build方法构建组件树。build()方法需要传入两个参数:第一个参数是通过wrapBuilder()封装的全局@Builder方法;第二个参数是对应的@Builder方法所需的参数对象。如果@Builder方法不带参数或者存在默认参数,则build()的第二个参数可以省略。

class TextNodeController extends NodeController {private textNode: BuilderNode<[Params]> | null = null;private message: string = '';constructor(message: string) {super();this.message = message;}makeNode(context: UIContext): FrameNode | null {this.textNode = new BuilderNode(context);this.textNode.build(wrapBuilder<[Params]>(buildText), new Params(this.message));return this.textNode.getFrameNode();}
}
5、显示自定义节点


显示自定义节点依赖于声明式渲染容器NodeContainer和对应的控制类NodeController。NodeController的makeNode方法返回的节点会显示在对应的NodeContainer中。由于makeNode需要返回一个FrameNode,因此如果预期显示BuilderNode,需要调用BuilderNode的getFrameNode方法来获取其根节点。详细代码如下:

@Entry
@Component
struct Index {@State message: string = "hello";private textNodeController: TextNodeController = new TextNodeController(this.message);build() {Row() {Column() {NodeContainer(this.textNodeController).width('100%').height(100).backgroundColor('#FFF0F0F0')}.width('100%').height('100%')}.height('100%')}
}
6、更新自定义节点


更新自定义节点可以参考BuilderNode的update方法。


http://www.ppmy.cn/ops/157010.html

相关文章

ES6-代码编程风格(数组、函数)

1 数组 使用扩展运算符&#xff08;...&#xff09;复制数组。 const itemsCopy [...items]; 使用Array.from 方法将类似数组的对象转为数组。 const foo document.querySelectorAll(.foo); const nodes Array.from(foo); 2 函数 立即执行函数可以写成箭头函数的形式…

SpringMVC常用的注解

Spring MVC 提供了丰富的注解&#xff0c;这些注解可以简化开发过程&#xff0c;提高开发效率&#xff0c;使代码结构更加清晰。以下是 Spring MVC 中一些常用注解的详细介绍&#xff1a; 1. 控制器相关注解 1.1 Controller 作用&#xff1a;用于标记一个类为控制器类&#…

TypeScript 中的接口:定义对象的形状

&#x1f90d; 前端开发工程师、技术日更博主、已过CET6 &#x1f368; 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 &#x1f560; 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》 &#x1f35a; 蓝桥云课签约作者、上架课程《Vue.js 和 E…

使用自定义大模型来部署Wren AI(开源的文本生成SQL方案)

使用自定义大模型来部署Wren AI&#xff08;开源的文本生成SQL方案&#xff09; 关于 首次发表日期&#xff1a;2024-07-15Wren AI官方文档&#xff1a; https://docs.getwren.ai/overview/introductionWren AI Github仓库&#xff1a; https://github.com/Canner/WrenAI 关…

网络安全-防御 第一次作业(由于防火墙只成功启动了一次未补截图)

防火墙安全策略课堂实验报告 一、拓扑 本实验拓扑包含预启动设备、DMZ区域&#xff08;含OA Server和Web Server&#xff09;、防火墙&#xff08;FW1&#xff09;、Trust区域&#xff08;含办公区PC和生产区PC&#xff09;等。具体IP地址及连接关系如给定拓扑图所示&#xf…

防洪子堤,筑牢生命防线|鼎跃安全

近年来&#xff0c;随着极端天气事件的增多和城市快速扩张&#xff0c;洪涝灾害频发已成为各级政府和社会各界普遍关注的问题。在抗洪抢险过程中&#xff0c;如何快速构筑防洪屏障、分流洪水、保护重点区域成为抢险救灾的关键。防洪子堤作为一种新型、灵活且易于部署的临时防洪…

如何利用Java爬虫获取商品销量详情实战指南

在当今数字化时代&#xff0c;电商平台的商品销量数据对于市场分析、竞品研究和商业决策具有极高的价值。通过Java爬虫技术&#xff0c;我们可以高效地获取这些数据&#xff0c;为商业分析提供支持。本文将详细介绍如何利用Java编写爬虫程序&#xff0c;获取商品的销量详情&…

Windows Docker笔记-简介摘录

Docker是一个开源的容器化平台&#xff0c;可以帮助开发人员将应用程序与其依赖项打包在一个独立的容器中&#xff0c;然后在任何安装的Docker的环境中快速、可靠地运行。 几个基本概念和优势&#xff1a; 1. 容器 容器是一个轻量级、独立的运行环境&#xff0c;包含了应用程…