1# 自定义组件冻结功能 2 3自定义组件冻结功能专为优化复杂UI页面的性能而设计,尤其适用于包含多个页面栈、长列表或宫格布局的场景。在这些情况下,当状态变量绑定了多个UI组件,其变化可能触发大量UI组件的刷新,进而导致界面卡顿和响应延迟。为了提升这类负载UI界面的刷新性能,开发者可以选择尝试使用自定义组件冻结功能。 4 5组件冻结的工作原理是: 61. 开发者通过设置freezeWhenInactive属性,即可激活组件冻结机制。 72. 启用后,系统将仅对处于激活状态的自定义组件进行更新,这使得UI框架可以尽量缩小更新范围,仅限于用户可见范围内(激活状态)的自定义组件,从而提高复杂UI场景下的刷新效率。 83. 当之前处于inactive状态的自定义组件重新变为active状态时,状态管理框架会对其执行必要的刷新操作,确保UI的正确展示。 9 10简而言之,组件冻结旨在优化复杂界面下的UI刷新性能。在存在多个不可见自定义组件的情况下,如多页面栈、长列表或宫格,通过组件冻结可以实现按需刷新,即仅刷新当前可见的自定义组件,而将不可见自定义组件的刷新延迟至它们变为可见时。 11 12需要注意,组件active/inactive并不等同于其可见性。组件冻结目前仅适用于以下场景: 13 141. 页面路由:当前栈顶页面为active状态,非栈顶不可见页面为inactive状态。 152. TabContent:只有当前显示的TabContent中的自定义组件处于active状态,其余则为inactive。 163. LazyForEach:仅当前显示的LazyForEach中的自定义组件为active状态,而缓存节点的组件则为inactive状态。 174. Navigation:当前显示的NavDestination中的自定义组件为active状态,而其他未显示的NavDestination组件则为inactive状态。 185. 组件复用:进入复用池的组件为inactive状态,从复用池上树的节点为active状态。 19其他场景,如堆叠布局(Stack)下的被遮罩的组件,这些组件尽管不可见,但并不被视为inactive状态,因此不在组件冻结的适用范围内。 20 21 22在阅读本文档前,开发者需要了解自定义组件基本语法。建议提前阅读:[自定义组件](./arkts-create-custom-components.md)。 23 24> **说明:** 25> 26> 从API version 11开始,支持自定义组件冻结功能。 27 28## 当前支持的场景 29 30### 页面路由 31 32> **说明:** 33> 34> 本示例使用了router进行页面跳转,建议开发者使用组件导航(Navigation)代替页面路由(router)来实现页面切换。Navigation提供了更多的功能和更灵活的自定义能力。请参考[使用Navigation的组件冻结用例](#navigation)。 35 36当页面1调用router.pushUrl接口跳转到页面2时,页面1为隐藏不可见状态,此时如果更新页面1中的状态变量,不会触发页面1刷新。 37图示如下: 38 39 40 41页面1: 42 43```ts 44import { router } from '@kit.ArkUI'; 45 46@Entry 47@Component({ freezeWhenInactive: true }) 48struct Page1 { 49 @StorageLink('PropA') @Watch("first") storageLink: number = 47; 50 51 first() { 52 console.info("first page " + `${this.storageLink}`) 53 } 54 55 build() { 56 Column() { 57 Text(`From first Page ${this.storageLink}`).fontSize(50) 58 Button('first page storageLink + 1').fontSize(30) 59 .onClick(() => { 60 this.storageLink += 1 61 }) 62 Button('go to next page').fontSize(30) 63 .onClick(() => { 64 router.pushUrl({ url: 'pages/Page2' }) 65 }) 66 } 67 } 68} 69``` 70 71页面2: 72 73```ts 74import { router } from '@kit.ArkUI'; 75 76@Entry 77@Component({ freezeWhenInactive: true }) 78struct Page2 { 79 @StorageLink('PropA') @Watch("second") storageLink2: number = 1; 80 81 second() { 82 console.info("second page: " + `${this.storageLink2}`) 83 } 84 85 build() { 86 Column() { 87 88 Text(`second Page ${this.storageLink2}`).fontSize(50) 89 Button('Change Divider.strokeWidth') 90 .onClick(() => { 91 router.back() 92 }) 93 94 Button('second page storageLink2 + 2').fontSize(30) 95 .onClick(() => { 96 this.storageLink2 += 2 97 }) 98 99 } 100 } 101} 102``` 103 104在上面的示例中: 105 1061.点击页面1中的Button “first page storageLink + 1”,storageLink状态变量改变,@Watch中注册的方法first会被调用。 107 1082.通过router.pushUrl({url: 'pages/second'}),跳转到页面2,页面1隐藏,状态由active变为inactive。 109 1103.点击页面2中的Button “this.storageLink2 += 2”,只回调页面2@Watch中注册的方法second,因为页面1的状态变量此时已被冻结。 111 1124.点击“back”,页面2被销毁,页面1的状态由inactive变为active,重新刷新在inactive时被冻结的状态变量,页面1@Watch中注册的方法first被再次调用。 113 114 115### TabContent 116 117- 对Tabs中当前不可见的TabContent进行冻结,不会触发组件的更新。 118 119- 需要注意的是:在首次渲染的时候,Tab只会创建当前正在显示的TabContent,当切换全部的TabContent后,TabContent才会被全部创建。 120 121图示如下: 122 123 124```ts 125@Entry 126@Component 127struct TabContentTest { 128 @State @Watch("onMessageUpdated") message: number = 0; 129 private data: number[] = [0, 1] 130 131 onMessageUpdated() { 132 console.info(`TabContent message callback func ${this.message}`) 133 } 134 135 build() { 136 Row() { 137 Column() { 138 Button('change message').onClick(() => { 139 this.message++ 140 }) 141 142 Tabs() { 143 ForEach(this.data, (item: number) => { 144 TabContent() { 145 FreezeChild({ message: this.message, index: item }) 146 }.tabBar(`tab${item}`) 147 }, (item: number) => item.toString()) 148 } 149 } 150 .width('100%') 151 } 152 .height('100%') 153 } 154} 155 156@Component({ freezeWhenInactive: true }) 157struct FreezeChild { 158 @Link @Watch("onMessageUpdated") message: number 159 private index: number = 0 160 161 onMessageUpdated() { 162 console.info(`FreezeChild message callback func ${this.message}, index: ${this.index}`) 163 } 164 165 build() { 166 Text("message" + `${this.message}, index: ${this.index}`) 167 .fontSize(50) 168 .fontWeight(FontWeight.Bold) 169 } 170} 171``` 172 173在上面的示例中: 174 1751.点击“change message”更改message的值,当前正在显示的TabContent组件中的@Watch中注册的方法onMessageUpdated被触发。 176 1772.点击“two”切换到另外的TabContent,TabContent状态由inactive变为active,对应的@Watch中注册的方法onMessageUpdated被触发。 178 1793.再次点击“change message”更改message的值,仅当前显示的TabContent子组件中的@Watch中注册的方法onMessageUpdated被触发。 180 181 182 183 184### LazyForEach 185 186- 对LazyForEach中缓存的自定义组件进行冻结,不会触发组件的更新。 187 188```ts 189// 用于处理数据监听的IDataSource的基本实现 190class BasicDataSource implements IDataSource { 191 private listeners: DataChangeListener[] = []; 192 private originDataArray: string[] = []; 193 194 public totalCount(): number { 195 return 0; 196 } 197 198 public getData(index: number): string { 199 return this.originDataArray[index]; 200 } 201 202 // 该方法为框架侧调用,为LazyForEach组件向其数据源处添加listener监听 203 registerDataChangeListener(listener: DataChangeListener): void { 204 if (this.listeners.indexOf(listener) < 0) { 205 console.info('add listener'); 206 this.listeners.push(listener); 207 } 208 } 209 210 // 该方法为框架侧调用,为对应的LazyForEach组件在数据源处去除listener监听 211 unregisterDataChangeListener(listener: DataChangeListener): void { 212 const pos = this.listeners.indexOf(listener); 213 if (pos >= 0) { 214 console.info('remove listener'); 215 this.listeners.splice(pos, 1); 216 } 217 } 218 219 // 通知LazyForEach组件需要重载所有子组件 220 notifyDataReload(): void { 221 this.listeners.forEach(listener => { 222 listener.onDataReloaded(); 223 }) 224 } 225 226 // 通知LazyForEach组件需要在index对应索引处添加子组件 227 notifyDataAdd(index: number): void { 228 this.listeners.forEach(listener => { 229 listener.onDataAdd(index); 230 }) 231 } 232 233 // 通知LazyForEach组件在index对应索引处数据有变化,需要重建该子组件 234 notifyDataChange(index: number): void { 235 this.listeners.forEach(listener => { 236 listener.onDataChange(index); 237 }) 238 } 239 240 // 通知LazyForEach组件需要在index对应索引处删除该子组件 241 notifyDataDelete(index: number): void { 242 this.listeners.forEach(listener => { 243 listener.onDataDelete(index); 244 }) 245 } 246} 247 248class MyDataSource extends BasicDataSource { 249 private dataArray: string[] = []; 250 251 public totalCount(): number { 252 return this.dataArray.length; 253 } 254 255 public getData(index: number): string { 256 return this.dataArray[index]; 257 } 258 259 public addData(index: number, data: string): void { 260 this.dataArray.splice(index, 0, data); 261 this.notifyDataAdd(index); 262 } 263 264 public pushData(data: string): void { 265 this.dataArray.push(data); 266 this.notifyDataAdd(this.dataArray.length - 1); 267 } 268} 269 270@Entry 271@Component 272struct LforEachTest { 273 private data: MyDataSource = new MyDataSource(); 274 @State @Watch("onMessageUpdated") message: number = 0; 275 276 onMessageUpdated() { 277 console.info(`LazyforEach message callback func ${this.message}`) 278 } 279 280 aboutToAppear() { 281 for (let i = 0; i <= 20; i++) { 282 this.data.pushData(`Hello ${i}`) 283 } 284 } 285 286 build() { 287 Column() { 288 Button('change message').onClick(() => { 289 this.message++ 290 }) 291 List({ space: 3 }) { 292 LazyForEach(this.data, (item: string) => { 293 ListItem() { 294 FreezeChild({ message: this.message, index: item }) 295 } 296 }, (item: string) => item) 297 }.cachedCount(5).height(500) 298 } 299 300 } 301} 302 303@Component({ freezeWhenInactive: true }) 304struct FreezeChild { 305 @Link @Watch("onMessageUpdated") message: number; 306 private index: string = ""; 307 308 aboutToAppear() { 309 console.info(`FreezeChild aboutToAppear index: ${this.index}`) 310 } 311 312 onMessageUpdated() { 313 console.info(`FreezeChild message callback func ${this.message}, index: ${this.index}`) 314 } 315 316 build() { 317 Text("message" + `${this.message}, index: ${this.index}`) 318 .width('90%') 319 .height(160) 320 .backgroundColor(0xAFEEEE) 321 .textAlign(TextAlign.Center) 322 .fontSize(30) 323 .fontWeight(FontWeight.Bold) 324 } 325} 326``` 327 328在上面的示例中: 329 3301.点击“change message”更改message的值,当前正在显示的ListItem中的子组件@Watch中注册的方法onMessageUpdated被触发。缓存节点@Watch中注册的方法不会被触发。(如果不加组件冻结,当前正在显示的ListItem和cachecount缓存节点@Watch中注册的方法onMessageUpdated都会触发watch回调。) 331 3322.List区域外的ListItem滑动到List区域内,状态由inactive变为active,对应的@Watch中注册的方法onMessageUpdated被触发。 333 3343.再次点击“change message”更改message的值,仅有当前显示的ListItem中的子组件@Watch中注册的方法onMessageUpdated被触发。 335 336 337 338### Navigation 339 340- 当NavDestination不可见时,会对其子自定义组件设置成非激活态,不会触发组件的刷新。当返回该页面时,其子自定义组件重新恢复成激活态,触发@Watch回调进行刷新。 341 342- 在下面例子中,NavigationContentMsgStack会被设置成非激活态,将不再响应状态变量的变化,也不会触发组件刷新。 343 344```ts 345@Entry 346@Component 347struct MyNavigationTestStack { 348 @Provide('pageInfo') pageInfo: NavPathStack = new NavPathStack(); 349 @State @Watch("info") message: number = 0; 350 @State logNumber: number = 0; 351 352 info() { 353 console.info(`freeze-test MyNavigation message callback ${this.message}`); 354 } 355 356 @Builder 357 PageMap(name: string) { 358 if (name === 'pageOne') { 359 pageOneStack({ message: this.message, logNumber: this.logNumber }) 360 } else if (name === 'pageTwo') { 361 pageTwoStack({ message: this.message, logNumber: this.logNumber }) 362 } else if (name === 'pageThree') { 363 pageThreeStack({ message: this.message, logNumber: this.logNumber }) 364 } 365 } 366 367 build() { 368 Column() { 369 Button('change message') 370 .onClick(() => { 371 this.message++; 372 }) 373 Navigation(this.pageInfo) { 374 Column() { 375 Button('Next Page', { stateEffect: true, type: ButtonType.Capsule }) 376 .width('80%') 377 .height(40) 378 .margin(20) 379 .onClick(() => { 380 this.pageInfo.pushPath({ name: 'pageOne' }); //将name指定的NavDestination页面信息入栈 381 }) 382 } 383 }.title('NavIndex') 384 .navDestination(this.PageMap) 385 .mode(NavigationMode.Stack) 386 } 387 } 388} 389 390@Component 391struct pageOneStack { 392 @Consume('pageInfo') pageInfo: NavPathStack; 393 @State index: number = 1; 394 @Link message: number; 395 @Link logNumber: number; 396 397 build() { 398 NavDestination() { 399 Column() { 400 NavigationContentMsgStack({ message: this.message, index: this.index, logNumber: this.logNumber }) 401 Text("cur stack size:" + `${this.pageInfo.size()}`) 402 .fontSize(30) 403 .fontWeight(FontWeight.Bold) 404 Button('Next Page', { stateEffect: true, type: ButtonType.Capsule }) 405 .width('80%') 406 .height(40) 407 .margin(20) 408 .onClick(() => { 409 this.pageInfo.pushPathByName('pageTwo', null); 410 }) 411 Button('Back Page', { stateEffect: true, type: ButtonType.Capsule }) 412 .width('80%') 413 .height(40) 414 .margin(20) 415 .onClick(() => { 416 this.pageInfo.pop(); 417 }) 418 }.width('100%').height('100%') 419 }.title('pageOne') 420 .onBackPressed(() => { 421 this.pageInfo.pop(); 422 return true; 423 }) 424 } 425} 426 427@Component 428struct pageTwoStack { 429 @Consume('pageInfo') pageInfo: NavPathStack; 430 @State index: number = 2; 431 @Link message: number; 432 @Link logNumber: number; 433 434 build() { 435 NavDestination() { 436 Column() { 437 NavigationContentMsgStack({ message: this.message, index: this.index, logNumber: this.logNumber }) 438 Text("cur stack size:" + `${this.pageInfo.size()}`) 439 .fontSize(30) 440 .fontWeight(FontWeight.Bold) 441 Button('Next Page', { stateEffect: true, type: ButtonType.Capsule }) 442 .width('80%') 443 .height(40) 444 .margin(20) 445 .onClick(() => { 446 this.pageInfo.pushPathByName('pageThree', null); 447 }) 448 Button('Back Page', { stateEffect: true, type: ButtonType.Capsule }) 449 .width('80%') 450 .height(40) 451 .margin(20) 452 .onClick(() => { 453 this.pageInfo.pop(); 454 }) 455 }.width('100%').height('100%') 456 }.title('pageTwo') 457 .onBackPressed(() => { 458 this.pageInfo.pop(); 459 return true; 460 }) 461 } 462} 463 464@Component 465struct pageThreeStack { 466 @Consume('pageInfo') pageInfo: NavPathStack; 467 @State index: number = 3; 468 @Link message: number; 469 @Link logNumber: number; 470 471 build() { 472 NavDestination() { 473 Column() { 474 NavigationContentMsgStack({ message: this.message, index: this.index, logNumber: this.logNumber }) 475 Text("cur stack size:" + `${this.pageInfo.size()}`) 476 .fontSize(30) 477 .fontWeight(FontWeight.Bold) 478 Button('Next Page', { stateEffect: true, type: ButtonType.Capsule }) 479 .width('80%') 480 .height(40) 481 .margin(20) 482 .onClick(() => { 483 this.pageInfo.pushPathByName('pageOne', null); 484 }) 485 Button('Back Page', { stateEffect: true, type: ButtonType.Capsule }) 486 .width('80%') 487 .height(40) 488 .margin(20) 489 .onClick(() => { 490 this.pageInfo.pop(); 491 }) 492 }.width('100%').height('100%') 493 }.title('pageThree') 494 .onBackPressed(() => { 495 this.pageInfo.pop(); 496 return true; 497 }) 498 } 499} 500 501@Component({ freezeWhenInactive: true }) 502struct NavigationContentMsgStack { 503 @Link @Watch("info") message: number; 504 @Link index: number; 505 @Link logNumber: number; 506 507 info() { 508 console.info(`freeze-test NavigationContent message callback ${this.message}`); 509 console.info(`freeze-test ---- called by content ${this.index}`); 510 this.logNumber++; 511 } 512 513 build() { 514 Column() { 515 Text("msg:" + `${this.message}`) 516 .fontSize(30) 517 .fontWeight(FontWeight.Bold) 518 Text("log number:" + `${this.logNumber}`) 519 .fontSize(30) 520 .fontWeight(FontWeight.Bold) 521 } 522 } 523} 524``` 525 526在上面的示例中: 527 5281.点击“change message”更改message的值,当前正在显示的MyNavigationTestStack组件中的@Watch中注册的方法info被触发。 529 5302.点击“Next Page”切换到PageOne,创建pageOneStack节点。 531 5323.再次点击“change message”更改message的值,仅pageOneStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。 533 5344.再次点击“Next Page”切换到PageTwo,创建pageTwoStack节点。 535 5365.再次点击“change message”更改message的值,仅pageTwoStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。 537 5386.再次点击“Next Page”切换到PageThree,创建pageThreeStack节点。 539 5407.再次点击“change message”更改message的值,仅pageThreeStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。 541 5428.点击“Back Page”回到PageTwo,此时,仅pageTwoStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。 543 5449.再次点击“Back Page”回到PageOne,此时,仅pageOneStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。 545 54610.再次点击“Back Page”回到初始页,此时,无任何触发。 547 548 549 550### 组件复用 551 552<!--RP1-->[组件复用](../performance/component-recycle.md)<!--RP1End-->通过重利用缓存池中已存在的节点,而非创建新节点,来优化UI性能并提升应用流畅度。复用池中的节点尽管未在UI组件树上展示,但是状态变量的更改仍会触发UI刷新。为了解决复用池中组件异常刷新问题,可以使用组件冻结避免复用池中的组件刷新。 553 554#### 组件复用、if和组件冻结混用场景 555下面是组件复用、if组件和组件冻结混合使用场景的例子,if组件绑定的状态变量变化成false时,触发子组件`ChildComponent`的下树,由于`ChildComponent`被标记了组件复用,所以不会被销毁,而是进入复用池,这个时候如果同时开启了组件冻结,则可以使在复用池里不再刷新。 556具体流程如下: 5571. 点击`change flag`,改变`flag`为false: 558 - 被标记\@Reusable的`ChildComponent`组件在下树时,不会被销毁,而是进入复用池,触发aboutToRecycle生命周期,同时设置状态为inactive。 559 - `ChildComponent`同时也开启了组件冻结,当其状态为inactive时,不会响应任何状态变量变化带来的UI刷新。 5602. 点击`change desc`,触发`Page`的成员变量`desc`的变化。 561 - `desc`是\@State装饰的,其变化会通知给其子组件`ChildComponent`\@Link装饰的`desc`。 562 - 但因为`ChildComponent`是inactive状态,且开启了组件冻结,所以这次变化并不会触发`@Watch('descChange')`的回调,以及`ChildComponent`UI刷新。如果没有开启组件冻结,当前`@Watch('descChange')`会立即回调,且复用池内的`ChildComponent`组件也会对应刷新。 5633. 再次点击`change flag`,改变`flag`为true: 564 - `ChildComponent`从复用池中重新加入到组件树上。 565 - 回调aboutToReuse生命周期,将当前最新的`count`值同步给子组件。`desc`是通过@State->@Link同步的,所以无需开发者手动在aboutToReuse中赋值。 566 - 设置ChildComponent为active状态,并且刷新在inactive时没有刷新的组件,在当前例子中,就是Text(ChildComponent desc: ${this.desc})。 567 568 569```ts 570@Reusable 571@Component({freezeWhenInactive: true}) 572struct ChildComponent { 573 @Link @Watch('descChange') desc: string; 574 @State count: number = 0; 575 descChange() { 576 console.info(`ChildComponent messageChange ${this.desc}`); 577 } 578 579 aboutToReuse(params: Record<string, ESObject>): void { 580 this.count = params.count as number; 581 } 582 583 aboutToRecycle(): void { 584 console.info(`ChildComponent has been recycled`); 585 } 586 build() { 587 Column() { 588 Text(`ChildComponent desc: ${this.desc}`) 589 .fontSize(20) 590 Text(`ChildComponent count ${this.count}`) 591 .fontSize(20) 592 }.border({width: 2, color: Color.Pink}) 593 } 594} 595 596@Entry 597@Component 598struct Page { 599 @State desc: string = 'Hello World'; 600 @State flag: boolean = true; 601 @State count: number = 0; 602 build() { 603 Column() { 604 Button(`change desc`).onClick(() => { 605 this.desc += '!'; 606 }) 607 Button(`change flag`).onClick(() => { 608 this.count++; 609 this.flag =! this.flag; 610 }) 611 if (this.flag) { 612 ChildComponent({desc: this.desc, count: this.count}) 613 } 614 } 615 .height('100%') 616 } 617} 618``` 619#### LazyForEach、组件复用和组件冻结混用场景 620在数据很多的长列表滑动场景下,开发者会使用LazyForEach来按需创建组件,同时配合组件复用降低在滑动过程中因创建和销毁组件带来的开销。 621但是开发者如果根据其复用类型不同,设置了<!--RP2-->[reuseId](../performance/component-recycle.md#接口说明)<!--RP2End-->,或者为了保证滑动性能设置了较大的cacheCount,这就可能使复用池或者LazyForEach缓存较多的节点。 622在这种情况下,如果开发者触发List下所有子节点的刷新,就会带来节点刷新数量过大的问题,这个时候,可以考虑搭配组件冻结使用。 623 624如下面例子: 6251. 滑动到index为14的位置,当前屏幕上可见区域内有15个`ChildComponent`。 6262. 在滑动过程中: 627 - 列表上端的`ChildComponent`滑出可视区域外,此时先进入LazyForEach的缓存区域内,被设置inactive。在滑出LazyForEach区域外后,因为标记了组件复用,所以并不会被析构,会进入复用池,此时再次被设置inactive。 628 - 列表下端LazyForEach的缓存节点会进入List范围内,此时会试图请求创建新的节点进入LazyForEach的缓存,发现有可复用的节点时,从复用池中拿出已有节点,触发aboutToReuse生命周期回调,此时因为节点进入的是LazyForEach的缓存区域,所以其状态依旧是inactive。 6293. 点击`change desc`,触发`Page`的成员变量`desc`的变化。 630 - `desc`是\@State装饰的,其变化会通知给其子组件`ChildComponent`\@Link装饰的`desc`。 631 - 非可视区域内的`ChildComponent`是inactive状态,且开启了组件冻结,所以这次变化只触发可视区域内的15个节点的`@Watch('descChange')`回调,并只刷新对应可视区域内的15个节点。LazyForEach和复用池中的节点并不会刷新,也不会触发\@Watch回调。 632 633图示如下: 634 635可通过trace观察,仅触发了15个`ChildComponent`节点的刷新。 636 637完整示例如下: 638```ts 639import { hiTraceMeter } from '@kit.PerformanceAnalysisKit'; 640// 用于处理数据监听的IDataSource的基本实现 641class BasicDataSource implements IDataSource { 642 private listeners: DataChangeListener[] = []; 643 private originDataArray: string[] = []; 644 645 public totalCount(): number { 646 return 0; 647 } 648 649 public getData(index: number): string { 650 return this.originDataArray[index]; 651 } 652 653 // 该方法为框架侧调用,为LazyForEach组件向其数据源处添加listener监听 654 registerDataChangeListener(listener: DataChangeListener): void { 655 if (this.listeners.indexOf(listener) < 0) { 656 console.info('add listener'); 657 this.listeners.push(listener); 658 } 659 } 660 661 // 该方法为框架侧调用,为对应的LazyForEach组件在数据源处去除listener监听 662 unregisterDataChangeListener(listener: DataChangeListener): void { 663 const pos = this.listeners.indexOf(listener); 664 if (pos >= 0) { 665 console.info('remove listener'); 666 this.listeners.splice(pos, 1); 667 } 668 } 669 670 // 通知LazyForEach组件需要重载所有子组件 671 notifyDataReload(): void { 672 this.listeners.forEach(listener => { 673 listener.onDataReloaded(); 674 }) 675 } 676 677 // 通知LazyForEach组件需要在index对应索引处添加子组件 678 notifyDataAdd(index: number): void { 679 this.listeners.forEach(listener => { 680 listener.onDataAdd(index); 681 }) 682 } 683 684 // 通知LazyForEach组件在index对应索引处数据有变化,需要重建该子组件 685 notifyDataChange(index: number): void { 686 this.listeners.forEach(listener => { 687 listener.onDataChange(index); 688 }) 689 } 690 691 // 通知LazyForEach组件需要在index对应索引处删除该子组件 692 notifyDataDelete(index: number): void { 693 this.listeners.forEach(listener => { 694 listener.onDataDelete(index); 695 }) 696 } 697 698 // 通知LazyForEach组件将from索引和to索引处的子组件进行交换 699 notifyDataMove(from: number, to: number): void { 700 this.listeners.forEach(listener => { 701 listener.onDataMove(from, to); 702 }) 703 } 704} 705 706class MyDataSource extends BasicDataSource { 707 private dataArray: string[] = []; 708 709 public totalCount(): number { 710 return this.dataArray.length; 711 } 712 713 public getData(index: number): string { 714 return this.dataArray[index]; 715 } 716 717 public addData(index: number, data: string): void { 718 this.dataArray.splice(index, 0, data); 719 this.notifyDataAdd(index); 720 } 721 722 public pushData(data: string): void { 723 this.dataArray.push(data); 724 this.notifyDataAdd(this.dataArray.length - 1); 725 } 726} 727 728@Reusable 729@Component({freezeWhenInactive: true}) 730struct ChildComponent { 731 @Link @Watch('descChange') desc: string; 732 @State item: string = ''; 733 @State index: number = 0; 734 descChange() { 735 console.info(`ChildComponent messageChange ${this.desc}`); 736 } 737 738 aboutToReuse(params: Record<string, ESObject>): void { 739 this.item = params.item; 740 this.index = params.index; 741 } 742 743 aboutToRecycle(): void { 744 console.info(`ChildComponent has been recycled`); 745 } 746 build() { 747 Column() { 748 Text(`ChildComponent index: ${this.index} item: ${this.item}`) 749 .fontSize(20) 750 Text(`desc: ${this.desc}`) 751 .fontSize(20) 752 }.border({width: 2, color: Color.Pink}) 753 } 754} 755 756@Entry 757@Component 758struct Page { 759 @State desc: string = 'Hello World'; 760 private data: MyDataSource = new MyDataSource(); 761 762 aboutToAppear() { 763 for (let i = 0; i < 50; i++) { 764 this.data.pushData(`Hello ${i}`); 765 } 766 } 767 768 build() { 769 Column() { 770 Button(`change desc`).onClick(() => { 771 hiTraceMeter.startTrace('change decs', 1); 772 this.desc += '!'; 773 hiTraceMeter.finishTrace('change decs', 1); 774 }) 775 List({ space: 3 }) { 776 LazyForEach(this.data, (item: string, index: number) => { 777 ListItem() { 778 ChildComponent({index: index, item: item, desc: this.desc}).reuseId(index % 10 < 5 ? "1": "0") 779 } 780 }, (item: string) => item) 781 }.cachedCount(5) 782 } 783 .height('100%') 784 } 785} 786``` 787#### LazyForEach、if、组件复用和组件冻结混用场景 788 789下面的场景中展示了LazyForEach、if、组件复用和组件冻结混用场景。在同一个父自定义组件下,可复用的节点可能通过不同的方式进入复用池,比如: 790- 通过滑动从LazyForEach的缓存区域下树,进入复用池。 791- if条件切换通知子节点下树,进入复用池。 792 793在下面的例子中: 7941. 当滑动到index为14的位置,屏幕上可见区域内有10个`ChildComponent`,9个是LazyForEach的子节点,1个是if的子节点。 7952. 点击`change flag`,if的条件变成false,其子节点`ChildComponent`进入复用池。当前屏幕显示9个节点。 7963. 此时不管是通过LazyForEach还是if下树的节点都会进入`Page`节点下的复用池。 7974. 点击`change desc`,仅更新屏幕上的9个`ChildComponent`节点,具体可参考下面的trace。 7985. 再次点击`change flag`,if的条件变成true,`ChildComponent`从复用池中重新加入到组件树上,其状态变成active。 7996. 再次点击`change desc`,从复用池中通过if和LazyForEach上树的节点都可正常刷新。 800 801开启组件冻结trace: 802 803 804 805没有开启组件冻结trace: 806 807 808 809 810完整例子如下: 811``` 812import { hiTraceMeter } from '@kit.PerformanceAnalysisKit'; 813class BasicDataSource implements IDataSource { 814 private listeners: DataChangeListener[] = []; 815 private originDataArray: string[] = []; 816 817 public totalCount(): number { 818 return 0; 819 } 820 821 public getData(index: number): string { 822 return this.originDataArray[index]; 823 } 824 825 // 该方法为框架侧调用,为LazyForEach组件向其数据源处添加listener监听 826 registerDataChangeListener(listener: DataChangeListener): void { 827 if (this.listeners.indexOf(listener) < 0) { 828 console.info('add listener'); 829 this.listeners.push(listener); 830 } 831 } 832 833 // 该方法为框架侧调用,为对应的LazyForEach组件在数据源处去除listener监听 834 unregisterDataChangeListener(listener: DataChangeListener): void { 835 const pos = this.listeners.indexOf(listener); 836 if (pos >= 0) { 837 console.info('remove listener'); 838 this.listeners.splice(pos, 1); 839 } 840 } 841 842 // 通知LazyForEach组件需要重载所有子组件 843 notifyDataReload(): void { 844 this.listeners.forEach(listener => { 845 listener.onDataReloaded(); 846 }) 847 } 848 849 // 通知LazyForEach组件需要在index对应索引处添加子组件 850 notifyDataAdd(index: number): void { 851 this.listeners.forEach(listener => { 852 listener.onDataAdd(index); 853 }) 854 } 855 856 // 通知LazyForEach组件在index对应索引处数据有变化,需要重建该子组件 857 notifyDataChange(index: number): void { 858 this.listeners.forEach(listener => { 859 listener.onDataChange(index); 860 }) 861 } 862 863 // 通知LazyForEach组件需要在index对应索引处删除该子组件 864 notifyDataDelete(index: number): void { 865 this.listeners.forEach(listener => { 866 listener.onDataDelete(index); 867 }) 868 } 869 870 // 通知LazyForEach组件将from索引和to索引处的子组件进行交换 871 notifyDataMove(from: number, to: number): void { 872 this.listeners.forEach(listener => { 873 listener.onDataMove(from, to); 874 }) 875 } 876} 877 878class MyDataSource extends BasicDataSource { 879 private dataArray: string[] = []; 880 881 public totalCount(): number { 882 return this.dataArray.length; 883 } 884 885 public getData(index: number): string { 886 return this.dataArray[index]; 887 } 888 889 public addData(index: number, data: string): void { 890 this.dataArray.splice(index, 0, data); 891 this.notifyDataAdd(index); 892 } 893 894 public pushData(data: string): void { 895 this.dataArray.push(data); 896 this.notifyDataAdd(this.dataArray.length - 1); 897 } 898} 899 900@Reusable 901@Component({freezeWhenInactive: true}) 902struct ChildComponent { 903 @Link @Watch('descChange') desc: string; 904 @State item: string = ''; 905 @State index: number = 0; 906 descChange() { 907 console.info(`ChildComponent messageChange ${this.desc}`); 908 } 909 910 aboutToReuse(params: Record<string, ESObject>): void { 911 this.item = params.item; 912 this.index = params.index; 913 } 914 915 aboutToRecycle(): void { 916 console.info(`ChildComponent has been recycled`); 917 } 918 build() { 919 Column() { 920 Text(`ChildComponent index: ${this.index} item: ${this.item}`) 921 .fontSize(20) 922 Text(`desc: ${this.desc}`) 923 .fontSize(20) 924 }.border({width: 2, color: Color.Pink}) 925 } 926} 927 928@Entry 929@Component 930struct Page { 931 @State desc: string = 'Hello World'; 932 @State flag: boolean = true; 933 private data: MyDataSource = new MyDataSource(); 934 935 aboutToAppear() { 936 for (let i = 0; i < 50; i++) { 937 this.data.pushData(`Hello ${i}`); 938 } 939 } 940 941 build() { 942 Column() { 943 Button(`change desc`).onClick(() => { 944 hiTraceMeter.startTrace('change decs', 1); 945 this.desc += '!'; 946 hiTraceMeter.finishTrace('change decs', 1); 947 }) 948 949 Button(`change flag`).onClick(() => { 950 hiTraceMeter.startTrace('change flag', 1); 951 this.flag = !this.flag; 952 hiTraceMeter.finishTrace('change flag', 1); 953 }) 954 955 List({ space: 3 }) { 956 LazyForEach(this.data, (item: string, index: number) => { 957 ListItem() { 958 ChildComponent({index: index, item: item, desc: this.desc}).reuseId(index % 10 < 5 ? "1": "0") 959 } 960 }, (item: string) => item) 961 } 962 .cachedCount(5) 963 .height('60%') 964 965 if (this.flag) { 966 ChildComponent({index: -1, item: 'Hello', desc: this.desc}).reuseId( "1") 967 } 968 } 969 .height('100%') 970 } 971} 972``` 973 974## 限制条件 975如下面的例子所示,FreezeBuildNode中使用了自定义节点[BuilderNode](../reference/apis-arkui/js-apis-arkui-builderNode.md)。BuilderNode可以通过命令式动态挂载组件,而组件冻结又是强依赖父子关系来通知是否开启组件冻结。如果父组件使用组件冻结,且组件树的中间层级上又启用了BuilderNode,则BuilderNode的子组件将无法被冻结。 976 977``` 978import { BuilderNode, FrameNode, NodeController, UIContext } from '@kit.ArkUI'; 979 980// 定义一个Params类,用于传递参数 981class Params { 982 index: number = 0; 983 984 constructor(index: number) { 985 this.index = index; 986 } 987} 988 989// 定义一个buildNodeChild组件,它包含一个message属性和一个index属性 990@Component 991struct buildNodeChild { 992 @StorageProp("buildNodeTest") @Watch("onMessageUpdated") message: string = "hello world"; 993 @State index: number = 0; 994 995 // 当message更新时,调用此方法 996 onMessageUpdated() { 997 console.log(`FreezeBuildNode builderNodeChild message callback func ${this.message},index:${this.index}`); 998 } 999 1000 build() { 1001 Text(`buildNode Child message: ${this.message}`).fontSize(30) 1002 } 1003} 1004 1005// 定义一个buildText函数,它接收一个Params参数并构建一个Column组件 1006@Builder 1007function buildText(params: Params) { 1008 Column() { 1009 buildNodeChild({ index: params.index }) 1010 } 1011} 1012 1013// 定义一个TextNodeController类,继承自NodeController 1014class TextNodeController extends NodeController { 1015 private textNode: BuilderNode<[Params]> | null = null; 1016 private index: number = 0; 1017 1018 // 构造函数接收一个index参数 1019 constructor(index: number) { 1020 super(); 1021 this.index = index; 1022 } 1023 1024 // 创建并返回一个FrameNode 1025 makeNode(context: UIContext): FrameNode | null { 1026 this.textNode = new BuilderNode(context); 1027 this.textNode.build(wrapBuilder<[Params]>(buildText), new Params(this.index)); 1028 return this.textNode.getFrameNode(); 1029 } 1030} 1031 1032// 定义一个Index组件,它包含一个message属性和一个data数组 1033@Entry 1034@Component 1035struct Index { 1036 @StorageLink("buildNodeTest") message: string = "hello"; 1037 private data: number[] = [0, 1]; 1038 1039 build() { 1040 Row() { 1041 Column() { 1042 Button("change").fontSize(30) 1043 .onClick(() => { 1044 this.message += 'a'; 1045 }) 1046 1047 Tabs() { 1048 ForEach(this.data, (item: number) => { 1049 TabContent() { 1050 FreezeBuildNode({ index: item }) 1051 }.tabBar(`tab${item}`) 1052 }, (item: number) => item.toString()) 1053 } 1054 } 1055 } 1056 .width('100%') 1057 .height('100%') 1058 } 1059} 1060 1061// 定义一个FreezeBuildNode组件,它包含一个message属性和一个index属性 1062@Component({ freezeWhenInactive: true }) 1063struct FreezeBuildNode { 1064 @StorageProp("buildNodeTest") @Watch("onMessageUpdated") message: string = "1111"; 1065 @State index: number = 0; 1066 1067 // 当message更新时,调用此方法 1068 onMessageUpdated() { 1069 console.log(`FreezeBuildNode message callback func ${this.message}, index: ${this.index}`); 1070 } 1071 1072 build() { 1073 NodeContainer(new TextNodeController(this.index)) 1074 .width('100%') 1075 .height('100%') 1076 .backgroundColor('#FFF0F0F0') 1077 } 1078} 1079``` 1080 1081在上面的示例中: 1082 1083点击Button("change")。改变message的值,当前正在显示的TabContent组件中的@Watch中注册的方法onMessageUpdated被触发。未显示的TabContent中的BuilderNode节点下组件的@Watch方法onMessageUpdated也被触发,并没有被冻结。 1084 1085 1086 1087