1# 自定义组件冻结功能 2<!--Kit: ArkUI--> 3<!--Subsystem: ArkUI--> 4<!--Owner: @liwenzhen3--> 5<!--Designer: @s10021109--> 6<!--Tester: @TerryTsao--> 7<!--Adviser: @zhang_yixin13--> 8 9自定义组件冻结功能专为优化复杂UI页面的性能而设计,尤其适用于包含多个页面栈、长列表或宫格布局的场景。当状态变量绑定多个UI组件时,其变化易触发大量组件刷新,导致界面卡顿与响应延迟。为提升这类高负载UI界面的刷新性能,建议开发者使用自定义组件冻结功能。 10 11组件冻结功能是一种性能优化机制,它会冻结非激活状态下的组件的刷新能力。当组件处于非激活状态时,即使其绑定的状态变量发生变化,也不会触发该组件的UI重新渲染,从而降低复杂UI场景下的刷新负载。 12 13组件冻结的工作原理是: 141. 开发者通过设置freezeWhenInactive属性,即可激活组件冻结机制。 152. 启用后,系统将仅对处于激活状态的自定义组件进行更新,这使得UI框架可以尽量缩小更新范围,仅限于用户可见范围内(激活状态)的自定义组件,从而提高复杂UI场景下的刷新效率。 163. 当之前处于inactive状态的自定义组件重新变为active状态时,状态管理框架会对其执行必要的刷新操作,确保UI的正确展示。 17 18简而言之,组件冻结旨在优化复杂界面下的UI刷新性能。在存在多个不可见自定义组件的情况下,如多页面栈、长列表或宫格,通过组件冻结可以实现按需刷新,即仅刷新当前可见的自定义组件,而将不可见自定义组件的刷新延迟至它们变为可见时。 19 20需要注意,组件active/inactive并不等同于其可见性。组件冻结目前仅适用于以下场景: 21 221. 页面路由:当前栈顶页面为active状态,非栈顶不可见页面为inactive状态。 232. TabContent:只有当前显示的TabContent中的自定义组件处于active状态,其余则为inactive。 243. LazyForEach:仅当前显示的LazyForEach中的自定义组件为active状态,而缓存节点的组件则为inactive状态。 254. Navigation:当前显示的NavDestination中的自定义组件为active状态,而其他未显示的NavDestination组件则为inactive状态。 265. 组件复用:进入复用池的组件为inactive状态,从复用池上树的节点为active状态。 276. 混用场景:对于以上场景的组合使用,例如TabContent下面使用LazyForEach,切换Tab时,API version 17及以下,LazyForEach中的所有节点都会被设置为active状态,而从API version 18开始,只有LazyForEach的屏上节点会被设置为active状态,其余则为inactive状态。 28 29在阅读本文档前,开发者需要了解自定义组件基本语法。建议提前阅读:[自定义组件](./arkts-create-custom-components.md)。 30 31> **说明:** 32> 33> 从API version 11开始,支持自定义组件冻结功能。 34> 35> 从API version 18开始,支持自定义组件冻结功能的混用场景冻结。 36 37## 当前支持的场景 38 39### 页面路由 40 41> **说明:** 42> 43> 本示例使用了router进行页面跳转,建议开发者使用组件导航(Navigation)代替页面路由(router)来实现页面切换。Navigation提供了更多的功能和更灵活的自定义能力。请参考[使用Navigation的组件冻结用例](#navigation)。 44 45当页面1调用router.pushUrl接口跳转到页面2时,页面1为隐藏不可见状态,此时如果更新页面1中的状态变量,不会触发页面1刷新。 46图示如下: 47 48 49 50页面1: 51 52```ts 53@Entry 54@Component({ freezeWhenInactive: true }) 55struct Page1 { 56 @StorageLink('PropA') @Watch('first') storageLink: number = 47; 57 58 first() { 59 console.info('first page ' + `${this.storageLink}`); 60 } 61 62 build() { 63 Column() { 64 Text(`From first Page ${this.storageLink}`).fontSize(50) 65 Button('first page storageLink + 1').fontSize(30) 66 .onClick(() => { 67 this.storageLink += 1; 68 }) 69 Button('go to next page').fontSize(30) 70 .onClick(() => { 71 this.getUIContext().getRouter().pushUrl({ url: 'pages/Page2' }); 72 }) 73 } 74 } 75} 76``` 77 78页面2: 79 80```ts 81@Entry 82@Component({ freezeWhenInactive: true }) 83struct Page2 { 84 @StorageLink('PropA') @Watch('second') storageLink2: number = 1; 85 86 second() { 87 console.info('second page: ' + `${this.storageLink2}`); 88 } 89 90 build() { 91 Column() { 92 93 Text(`second Page ${this.storageLink2}`).fontSize(50) 94 Button('Change Divider.strokeWidth') 95 .onClick(() => { 96 this.getUIContext().getRouter().back(); 97 }) 98 99 Button('second page storageLink2 + 2').fontSize(30) 100 .onClick(() => { 101 this.storageLink2 += 2; 102 }) 103 104 } 105 } 106} 107``` 108 109在上面的示例中: 110 1111.点击页面1中的Button “first page storageLink + 1”,storageLink状态变量改变,[@Watch](./arkts-watch.md)中注册的方法first会被调用。 112 1132.通过router.pushUrl({url: 'pages/second'}),跳转到页面2,页面1隐藏,状态由active变为inactive。 114 1153.点击页面2中的Button “this.storageLink2 += 2”,只回调页面2@Watch中注册的方法second,因为页面1的状态变量此时已被冻结。 116 1174.点击“back”,页面2被销毁,页面1的状态由inactive变为active,重新刷新在inactive时被冻结的状态变量,页面1@Watch中注册的方法first被再次调用。 118 119 120### TabContent 121 122- 对Tabs中当前不可见的TabContent进行冻结,不会触发组件的更新。 123 124- 需要注意的是:在首次渲染的时候,Tab只会创建当前正在显示的TabContent,当切换全部的TabContent后,TabContent才会被全部创建。 125 126图示如下: 127 128 129```ts 130@Entry 131@Component 132struct TabContentTest { 133 @State @Watch('onMessageUpdated') message: number = 0; 134 private data: number[] = [0, 1]; 135 136 onMessageUpdated() { 137 console.info(`TabContent message callback func ${this.message}`); 138 } 139 140 build() { 141 Row() { 142 Column() { 143 Button('change message').onClick(() => { 144 this.message++; 145 }) 146 147 Tabs() { 148 ForEach(this.data, (item: number) => { 149 TabContent() { 150 FreezeChild({ message: this.message, index: item }) 151 }.tabBar(`tab${item}`) 152 }, (item: number) => item.toString()) 153 } 154 } 155 .width('100%') 156 } 157 .height('100%') 158 } 159} 160 161@Component({ freezeWhenInactive: true }) 162struct FreezeChild { 163 @Link @Watch('onMessageUpdated') message: number; 164 index: number = 0; 165 166 onMessageUpdated() { 167 console.info(`FreezeChild message callback func ${this.message}, index: ${this.index}`); 168 } 169 170 build() { 171 Text('message' + `${this.message}, index: ${this.index}`) 172 .fontSize(50) 173 .fontWeight(FontWeight.Bold) 174 } 175} 176``` 177 178在上面的示例中: 179 1801.点击“change message”更改message的值,当前正在显示的TabContent组件中的@Watch中注册的方法onMessageUpdated被触发。 181 1822.点击“tab1”切换到另外的TabContent,TabContent状态由inactive变为active,对应的@Watch中注册的方法onMessageUpdated被触发。 183 1843.再次点击“change message”更改message的值,仅当前显示的TabContent子组件中的@Watch中注册的方法onMessageUpdated被触发。 185 186 187 188 189### LazyForEach 190 191- 对LazyForEach中缓存的自定义组件进行冻结,不会触发组件的更新。 192 193```ts 194// 用于处理数据监听的IDataSource的基本实现 195class BasicDataSource implements IDataSource { 196 private listeners: DataChangeListener[] = []; 197 private originDataArray: string[] = []; 198 199 public totalCount(): number { 200 return 0; 201 } 202 203 public getData(index: number): string { 204 return this.originDataArray[index]; 205 } 206 207 // 该方法为框架侧调用,为LazyForEach组件向其数据源处添加listener监听 208 registerDataChangeListener(listener: DataChangeListener): void { 209 if (this.listeners.indexOf(listener) < 0) { 210 console.info('add listener'); 211 this.listeners.push(listener); 212 } 213 } 214 215 // 该方法为框架侧调用,为对应的LazyForEach组件在数据源处去除listener监听 216 unregisterDataChangeListener(listener: DataChangeListener): void { 217 const pos = this.listeners.indexOf(listener); 218 if (pos >= 0) { 219 console.info('remove listener'); 220 this.listeners.splice(pos, 1); 221 } 222 } 223 224 // 通知LazyForEach组件需要重载所有子组件 225 notifyDataReload(): void { 226 this.listeners.forEach(listener => { 227 listener.onDataReloaded(); 228 }) 229 } 230 231 // 通知LazyForEach组件需要在index对应索引处添加子组件 232 notifyDataAdd(index: number): void { 233 this.listeners.forEach(listener => { 234 listener.onDataAdd(index); 235 }) 236 } 237 238 // 通知LazyForEach组件在index对应索引处数据有变化,需要重建该子组件 239 notifyDataChange(index: number): void { 240 this.listeners.forEach(listener => { 241 listener.onDataChange(index); 242 }) 243 } 244 245 // 通知LazyForEach组件需要在index对应索引处删除该子组件 246 notifyDataDelete(index: number): void { 247 this.listeners.forEach(listener => { 248 listener.onDataDelete(index); 249 }) 250 } 251} 252 253class MyDataSource extends BasicDataSource { 254 private dataArray: string[] = []; 255 256 public totalCount(): number { 257 return this.dataArray.length; 258 } 259 260 public getData(index: number): string { 261 return this.dataArray[index]; 262 } 263 264 public addData(index: number, data: string): void { 265 this.dataArray.splice(index, 0, data); 266 this.notifyDataAdd(index); 267 } 268 269 public pushData(data: string): void { 270 this.dataArray.push(data); 271 this.notifyDataAdd(this.dataArray.length - 1); 272 } 273} 274 275@Entry 276@Component 277struct LforEachTest { 278 private data: MyDataSource = new MyDataSource(); 279 @State @Watch('onMessageUpdated') message: number = 0; 280 281 onMessageUpdated() { 282 console.info(`LazyforEach message callback func ${this.message}`); 283 } 284 285 aboutToAppear() { 286 for (let i = 0; i <= 20; i++) { 287 this.data.pushData(`Hello ${i}`); 288 } 289 } 290 291 build() { 292 Column() { 293 Button('change message').onClick(() => { 294 this.message++; 295 }) 296 List({ space: 3 }) { 297 LazyForEach(this.data, (item: string) => { 298 ListItem() { 299 FreezeChild({ message: this.message, index: item }) 300 } 301 }, (item: string) => item) 302 }.cachedCount(5).height(500) 303 } 304 305 } 306} 307 308@Component({ freezeWhenInactive: true }) 309struct FreezeChild { 310 @Link @Watch('onMessageUpdated') message: number; 311 index: string = ''; 312 313 aboutToAppear() { 314 console.info(`FreezeChild aboutToAppear index: ${this.index}`); 315 } 316 317 onMessageUpdated() { 318 console.info(`FreezeChild message callback func ${this.message}, index: ${this.index}`); 319 } 320 321 build() { 322 Text('message' + `${this.message}, index: ${this.index}`) 323 .width('90%') 324 .height(160) 325 .backgroundColor(0xAFEEEE) 326 .textAlign(TextAlign.Center) 327 .fontSize(30) 328 .fontWeight(FontWeight.Bold) 329 } 330} 331``` 332 333在上面的示例中: 334 3351.点击“change message”更改message的值,当前正在显示的ListItem中的子组件@Watch中注册的方法onMessageUpdated被触发。缓存节点@Watch中注册的方法不会被触发。(如果不加组件冻结,当前正在显示的ListItem和cachecount缓存节点@Watch中注册的方法onMessageUpdated都会触发watch回调。) 336 3372.List区域外的ListItem滑动到List区域内,状态由inactive变为active,对应的@Watch中注册的方法onMessageUpdated被触发。 338 3393.再次点击“change message”更改message的值,仅有当前显示的ListItem中的子组件@Watch中注册的方法onMessageUpdated被触发。 340 341 342 343### Navigation 344 345- 当NavDestination不可见时,会将其子自定义组件设置成非激活态,不会触发组件的刷新。当返回该页面时,其子自定义组件重新恢复成激活态,触发@Watch回调进行刷新。 346 347- 在下面例子中,NavigationContentMsgStack会被设置成非激活态,将不再响应状态变量的变化,也不会触发组件刷新。 348 349```ts 350@Entry 351@Component 352struct MyNavigationTestStack { 353 @Provide('pageInfo') pageInfo: NavPathStack = new NavPathStack(); 354 @State @Watch('info') message: number = 0; 355 @State logNumber: number = 0; 356 357 info() { 358 console.info(`freeze-test MyNavigation message callback ${this.message}`); 359 } 360 361 @Builder 362 PageMap(name: string) { 363 if (name === 'pageOne') { 364 PageOneStack({ message: this.message, logNumber: this.logNumber }) 365 } else if (name === 'pageTwo') { 366 PageTwoStack({ message: this.message, logNumber: this.logNumber }) 367 } else if (name === 'pageThree') { 368 PageThreeStack({ message: this.message, logNumber: this.logNumber }) 369 } 370 } 371 372 build() { 373 Column() { 374 Button('change message') 375 .onClick(() => { 376 this.message++; 377 }) 378 Navigation(this.pageInfo) { 379 Column() { 380 Button('Next Page', { stateEffect: true, type: ButtonType.Capsule }) 381 .width('80%') 382 .height(40) 383 .margin(20) 384 .onClick(() => { 385 this.pageInfo.pushPath({ name: 'pageOne' }); //将name指定的NavDestination页面信息入栈 386 }) 387 } 388 }.title('NavIndex') 389 .navDestination(this.PageMap) 390 .mode(NavigationMode.Stack) 391 } 392 } 393} 394 395@Component 396struct PageOneStack { 397 @Consume('pageInfo') pageInfo: NavPathStack; 398 @State index: number = 1; 399 @Link message: number; 400 @Link logNumber: number; 401 402 build() { 403 NavDestination() { 404 Column() { 405 NavigationContentMsgStack({ message: this.message, index: this.index, logNumber: this.logNumber }) 406 Text('cur stack size:' + `${this.pageInfo.size()}`) 407 .fontSize(30) 408 .fontWeight(FontWeight.Bold) 409 Button('Next Page', { stateEffect: true, type: ButtonType.Capsule }) 410 .width('80%') 411 .height(40) 412 .margin(20) 413 .onClick(() => { 414 this.pageInfo.pushPathByName('pageTwo', null); 415 }) 416 Button('Back Page', { stateEffect: true, type: ButtonType.Capsule }) 417 .width('80%') 418 .height(40) 419 .margin(20) 420 .onClick(() => { 421 this.pageInfo.pop(); 422 }) 423 }.width('100%').height('100%') 424 }.title('pageOne') 425 .onBackPressed(() => { 426 this.pageInfo.pop(); 427 return true; 428 }) 429 } 430} 431 432@Component 433struct PageTwoStack { 434 @Consume('pageInfo') pageInfo: NavPathStack; 435 @State index: number = 2; 436 @Link message: number; 437 @Link logNumber: number; 438 439 build() { 440 NavDestination() { 441 Column() { 442 NavigationContentMsgStack({ message: this.message, index: this.index, logNumber: this.logNumber }) 443 Text('cur stack size:' + `${this.pageInfo.size()}`) 444 .fontSize(30) 445 .fontWeight(FontWeight.Bold) 446 Button('Next Page', { stateEffect: true, type: ButtonType.Capsule }) 447 .width('80%') 448 .height(40) 449 .margin(20) 450 .onClick(() => { 451 this.pageInfo.pushPathByName('pageThree', null); 452 }) 453 Button('Back Page', { stateEffect: true, type: ButtonType.Capsule }) 454 .width('80%') 455 .height(40) 456 .margin(20) 457 .onClick(() => { 458 this.pageInfo.pop(); 459 }) 460 }.width('100%').height('100%') 461 }.title('pageTwo') 462 .onBackPressed(() => { 463 this.pageInfo.pop(); 464 return true; 465 }) 466 } 467} 468 469@Component 470struct PageThreeStack { 471 @Consume('pageInfo') pageInfo: NavPathStack; 472 @State index: number = 3; 473 @Link message: number; 474 @Link logNumber: number; 475 476 build() { 477 NavDestination() { 478 Column() { 479 NavigationContentMsgStack({ message: this.message, index: this.index, logNumber: this.logNumber }) 480 Text('cur stack size:' + `${this.pageInfo.size()}`) 481 .fontSize(30) 482 .fontWeight(FontWeight.Bold) 483 Button('Next Page', { stateEffect: true, type: ButtonType.Capsule }) 484 .width('80%') 485 .height(40) 486 .margin(20) 487 .onClick(() => { 488 this.pageInfo.pushPathByName('pageOne', null); 489 }) 490 Button('Back Page', { stateEffect: true, type: ButtonType.Capsule }) 491 .width('80%') 492 .height(40) 493 .margin(20) 494 .onClick(() => { 495 this.pageInfo.pop(); 496 }) 497 }.width('100%').height('100%') 498 }.title('pageThree') 499 .onBackPressed(() => { 500 this.pageInfo.pop(); 501 return true; 502 }) 503 } 504} 505 506@Component({ freezeWhenInactive: true }) 507struct NavigationContentMsgStack { 508 @Link @Watch('info') message: number; 509 @Link index: number; 510 @Link logNumber: number; 511 512 info() { 513 console.info(`freeze-test NavigationContent message callback ${this.message}`); 514 console.info(`freeze-test ---- called by content ${this.index}`); 515 this.logNumber++; 516 } 517 518 build() { 519 Column() { 520 Text('msg:' + `${this.message}`) 521 .fontSize(30) 522 .fontWeight(FontWeight.Bold) 523 Text('log number:' + `${this.logNumber}`) 524 .fontSize(30) 525 .fontWeight(FontWeight.Bold) 526 } 527 } 528} 529``` 530 531在上面的示例中: 532 5331.点击“change message”更改message的值,当前正在显示的MyNavigationTestStack组件中的@Watch中注册的方法info被触发。 534 5352.点击“Next Page”切换到PageOne,创建PageOneStack节点。 536 5373.再次点击“change message”更改message的值,仅PageOneStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。 538 5394.再次点击“Next Page”切换到PageTwo,创建PageTwoStack节点。 540 5415.再次点击“change message”更改message的值,仅PageTwoStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。 542 5436.再次点击“Next Page”切换到PageThree,创建PageThreeStack节点。 544 5457.再次点击“change message”更改message的值,仅PageThreeStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。 546 5478.点击“Back Page”回到PageTwo,此时,仅PageTwoStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。 548 5499.再次点击“Back Page”回到PageOne,此时,仅PageOneStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。 550 55110.再次点击“Back Page”回到初始页,此时,无任何触发。 552 553 554 555### 组件复用 556 557[组件复用](./arkts-reusable.md)通过重利用缓存池中已存在的节点,而非创建新节点,来优化UI性能并提升应用流畅度。复用池中的节点尽管未在UI组件树上展示,但是状态变量的更改仍会触发UI刷新。为了解决复用池中组件异常刷新问题,可以使用组件冻结避免复用池中的组件刷新。 558 559**组件复用、if和组件冻结混用场景** 560 561下面是组件复用、if组件和组件冻结混合使用场景的例子,if组件绑定的状态变量变化成false时,触发子组件`ChildComponent`的下树,由于`ChildComponent`被标记了组件复用,所以不会被销毁,而是进入复用池,这个时候如果同时开启了组件冻结,则可以使在复用池里不再刷新。 562具体流程如下: 5631. 点击`change flag`,改变`flag`为false: 564 - 被标记\@Reusable的`ChildComponent`组件在下树时,不会被销毁,而是进入复用池,触发aboutToRecycle生命周期,同时设置状态为inactive。 565 - `ChildComponent`同时也开启了组件冻结,当其状态为inactive时,不会响应任何状态变量变化带来的UI刷新。 5662. 点击`change desc`,触发`Page`的成员变量`desc`的变化: 567 - `desc`是\@State装饰的,其变化会通知给其子组件`ChildComponent`[\@Link](./arkts-link.md)装饰的`desc`。 568 - 但因为`ChildComponent`是inactive状态,且开启了组件冻结,所以这次变化并不会触发`@Watch('descChange')`的回调,以及`ChildComponent`UI刷新。如果没有开启组件冻结,当前`@Watch('descChange')`会立即回调,且复用池内的`ChildComponent`组件也会对应刷新。 5693. 再次点击`change flag`,改变`flag`为true: 570 - `ChildComponent`从复用池中重新加入到组件树上。 571 - 回调aboutToReuse生命周期,将当前最新的`count`值同步给子组件。`desc`是通过[@State](./arkts-state.md)->@Link同步的,所以无需开发者手动在aboutToReuse中赋值。 572 - 设置ChildComponent为active状态,并且刷新在inactive时没有刷新的组件,在当前例子中,就是Text(ChildComponent desc: ${this.desc})。 573 574 575```ts 576@Reusable 577@Component({ freezeWhenInactive: true }) 578struct ChildComponent { 579 @Link @Watch('descChange') desc: string; 580 @State count: number = 0; 581 582 descChange() { 583 console.info(`ChildComponent messageChange ${this.desc}`); 584 } 585 586 aboutToReuse(params: Record<string, ESObject>): void { 587 this.count = params.count as number; 588 } 589 590 aboutToRecycle(): void { 591 console.info(`ChildComponent has been recycled`); 592 } 593 594 build() { 595 Column() { 596 Text(`ChildComponent desc: ${this.desc}`) 597 .fontSize(20) 598 Text(`ChildComponent count ${this.count}`) 599 .fontSize(20) 600 }.border({ width: 2, color: Color.Pink }) 601 } 602} 603 604@Entry 605@Component 606struct Page { 607 @State desc: string = 'Hello World'; 608 @State flag: boolean = true; 609 @State count: number = 0; 610 611 build() { 612 Column() { 613 Button(`change desc`).onClick(() => { 614 this.desc += '!'; 615 }) 616 Button(`change flag`).onClick(() => { 617 this.count++; 618 this.flag = !this.flag; 619 }) 620 if (this.flag) { 621 ChildComponent({ desc: this.desc, count: this.count }) 622 } 623 } 624 .height('100%') 625 } 626} 627``` 628**LazyForEach、组件复用和组件冻结混用场景** 629 630在数据很多的长列表滑动场景下,开发者会使用LazyForEach来按需创建组件,同时配合组件复用降低在滑动过程中因创建和销毁组件带来的开销。 631但是开发者如果根据其复用类型不同,设置了[reuseId](../../reference/apis-arkui/arkui-ts/ts-universal-attributes-reuse-id.md#reuseid),或者为了保证滑动性能设置了较大的cacheCount,这就可能使复用池或者LazyForEach缓存较多的节点。 632在这种情况下,如果开发者触发List下所有子节点的刷新,就会带来节点刷新数量过大的问题,这个时候,可以考虑搭配组件冻结使用。 633 634如下面例子: 6351. 滑动到index为14的位置,当前屏幕上可见区域内有15个`ChildComponent`。 6362. 在滑动过程中: 637 - 列表上端的`ChildComponent`滑出可视区域外,此时先进入LazyForEach的缓存区域内,被设置inactive。在滑出LazyForEach缓存区域外后,因为标记了组件复用,所以并不会被析构,而是会进入复用池,此时再次被设置inactive。 638 - 列表下端LazyForEach的缓存节点会进入List范围内,此时会试图请求创建新的节点进入LazyForEach的缓存,发现有可复用的节点时,从复用池中拿出已有节点,触发aboutToReuse生命周期回调,此时因为节点进入的是LazyForEach的缓存区域,所以其状态依旧是inactive。 6393. 点击`change desc`,触发`Page`的成员变量`desc`的变化: 640 - `desc`是\@State装饰的,其变化会通知给其子组件`ChildComponent`\@Link装饰的`desc`。 641 - 非可视区域内的`ChildComponent`是inactive状态,且开启了组件冻结,所以这次变化只触发可视区域内的15个节点的`@Watch('descChange')`回调,并只刷新对应可视区域内的15个节点。LazyForEach和复用池中的节点并不会刷新,也不会触发\@Watch回调。 642 643 644图示如下: 645 646可通过trace观察,仅触发了15个`ChildComponent`节点的刷新。 647 648完整示例如下: 649```ts 650import { hiTraceMeter } from '@kit.PerformanceAnalysisKit'; 651// 用于处理数据监听的IDataSource的基本实现 652class BasicDataSource implements IDataSource { 653 private listeners: DataChangeListener[] = []; 654 private originDataArray: string[] = []; 655 656 public totalCount(): number { 657 return 0; 658 } 659 660 public getData(index: number): string { 661 return this.originDataArray[index]; 662 } 663 664 // 该方法为框架侧调用,为LazyForEach组件向其数据源处添加listener监听 665 registerDataChangeListener(listener: DataChangeListener): void { 666 if (this.listeners.indexOf(listener) < 0) { 667 console.info('add listener'); 668 this.listeners.push(listener); 669 } 670 } 671 672 // 该方法为框架侧调用,为对应的LazyForEach组件在数据源处去除listener监听 673 unregisterDataChangeListener(listener: DataChangeListener): void { 674 const pos = this.listeners.indexOf(listener); 675 if (pos >= 0) { 676 console.info('remove listener'); 677 this.listeners.splice(pos, 1); 678 } 679 } 680 681 // 通知LazyForEach组件需要重载所有子组件 682 notifyDataReload(): void { 683 this.listeners.forEach(listener => { 684 listener.onDataReloaded(); 685 }) 686 } 687 688 // 通知LazyForEach组件需要在index对应索引处添加子组件 689 notifyDataAdd(index: number): void { 690 this.listeners.forEach(listener => { 691 listener.onDataAdd(index); 692 }) 693 } 694 695 // 通知LazyForEach组件在index对应索引处数据有变化,需要重建该子组件 696 notifyDataChange(index: number): void { 697 this.listeners.forEach(listener => { 698 listener.onDataChange(index); 699 }) 700 } 701 702 // 通知LazyForEach组件需要在index对应索引处删除该子组件 703 notifyDataDelete(index: number): void { 704 this.listeners.forEach(listener => { 705 listener.onDataDelete(index); 706 }) 707 } 708 709 // 通知LazyForEach组件将from索引和to索引处的子组件进行交换 710 notifyDataMove(from: number, to: number): void { 711 this.listeners.forEach(listener => { 712 listener.onDataMove(from, to); 713 }) 714 } 715} 716 717class MyDataSource extends BasicDataSource { 718 private dataArray: string[] = []; 719 720 public totalCount(): number { 721 return this.dataArray.length; 722 } 723 724 public getData(index: number): string { 725 return this.dataArray[index]; 726 } 727 728 public addData(index: number, data: string): void { 729 this.dataArray.splice(index, 0, data); 730 this.notifyDataAdd(index); 731 } 732 733 public pushData(data: string): void { 734 this.dataArray.push(data); 735 this.notifyDataAdd(this.dataArray.length - 1); 736 } 737} 738 739@Reusable 740@Component({freezeWhenInactive: true}) 741struct ChildComponent { 742 @Link @Watch('descChange') desc: string; 743 @State item: string = ''; 744 @State index: number = 0; 745 descChange() { 746 console.info(`ChildComponent messageChange ${this.desc}`); 747 } 748 749 aboutToReuse(params: Record<string, ESObject>): void { 750 this.item = params.item; 751 this.index = params.index; 752 } 753 754 aboutToRecycle(): void { 755 console.info(`ChildComponent has been recycled`); 756 } 757 build() { 758 Column() { 759 Text(`ChildComponent index: ${this.index} item: ${this.item}`) 760 .fontSize(20) 761 Text(`desc: ${this.desc}`) 762 .fontSize(20) 763 }.border({width: 2, color: Color.Pink}) 764 } 765} 766 767@Entry 768@Component 769struct Page { 770 @State desc: string = 'Hello World'; 771 private data: MyDataSource = new MyDataSource(); 772 773 aboutToAppear() { 774 for (let i = 0; i < 50; i++) { 775 this.data.pushData(`Hello ${i}`); 776 } 777 } 778 779 build() { 780 Column() { 781 Button(`change desc`).onClick(() => { 782 hiTraceMeter.startTrace('change desc', 1); 783 this.desc += '!'; 784 hiTraceMeter.finishTrace('change desc', 1); 785 }) 786 List({ space: 3 }) { 787 LazyForEach(this.data, (item: string, index: number) => { 788 ListItem() { 789 ChildComponent({index: index, item: item, desc: this.desc}).reuseId(index % 10 < 5 ? '1': '0') 790 } 791 }, (item: string) => item) 792 }.cachedCount(5) 793 } 794 .height('100%') 795 } 796} 797``` 798**LazyForEach、if、组件复用和组件冻结混用场景** 799 800下面的场景中展示了LazyForEach、if、组件复用和组件冻结混用场景。在同一个父自定义组件下,可复用的节点可能通过不同的方式进入复用池,比如: 801- 通过滑动从LazyForEach的缓存区域下树,进入复用池。 802- if条件切换通知子节点下树,进入复用池。 803 804在下面的例子中: 8051. 当滑动到index为14的位置,屏幕上可见区域内有10个`ChildComponent`,9个是LazyForEach的子节点,1个是if的子节点。 8062. 点击`change flag`,if的条件变成false,其子节点`ChildComponent`进入复用池。当前屏幕显示9个节点。 8073. 此时不管是通过LazyForEach还是if下树的节点都会进入`Page`节点下的复用池。 8084. 点击`change desc`,仅更新屏幕上的9个`ChildComponent`节点,具体可参考下面的trace。 8095. 再次点击`change flag`,if的条件变成true,`ChildComponent`从复用池中重新加入到组件树上,其状态变成active。 8106. 再次点击`change desc`,从复用池中通过if和LazyForEach上树的节点都可正常刷新。 811 812开启组件冻结trace: 813 814 815 816没有开启组件冻结trace: 817 818 819 820 821完整例子如下: 822``` 823import { hiTraceMeter } from '@kit.PerformanceAnalysisKit'; 824class BasicDataSource implements IDataSource { 825 private listeners: DataChangeListener[] = []; 826 private originDataArray: string[] = []; 827 828 public totalCount(): number { 829 return 0; 830 } 831 832 public getData(index: number): string { 833 return this.originDataArray[index]; 834 } 835 836 // 该方法为框架侧调用,为LazyForEach组件向其数据源处添加listener监听 837 registerDataChangeListener(listener: DataChangeListener): void { 838 if (this.listeners.indexOf(listener) < 0) { 839 console.info('add listener'); 840 this.listeners.push(listener); 841 } 842 } 843 844 // 该方法为框架侧调用,为对应的LazyForEach组件在数据源处去除listener监听 845 unregisterDataChangeListener(listener: DataChangeListener): void { 846 const pos = this.listeners.indexOf(listener); 847 if (pos >= 0) { 848 console.info('remove listener'); 849 this.listeners.splice(pos, 1); 850 } 851 } 852 853 // 通知LazyForEach组件需要重载所有子组件 854 notifyDataReload(): void { 855 this.listeners.forEach(listener => { 856 listener.onDataReloaded(); 857 }) 858 } 859 860 // 通知LazyForEach组件需要在index对应索引处添加子组件 861 notifyDataAdd(index: number): void { 862 this.listeners.forEach(listener => { 863 listener.onDataAdd(index); 864 }) 865 } 866 867 // 通知LazyForEach组件在index对应索引处数据有变化,需要重建该子组件 868 notifyDataChange(index: number): void { 869 this.listeners.forEach(listener => { 870 listener.onDataChange(index); 871 }) 872 } 873 874 // 通知LazyForEach组件需要在index对应索引处删除该子组件 875 notifyDataDelete(index: number): void { 876 this.listeners.forEach(listener => { 877 listener.onDataDelete(index); 878 }) 879 } 880 881 // 通知LazyForEach组件将from索引和to索引处的子组件进行交换 882 notifyDataMove(from: number, to: number): void { 883 this.listeners.forEach(listener => { 884 listener.onDataMove(from, to); 885 }) 886 } 887} 888 889class MyDataSource extends BasicDataSource { 890 private dataArray: string[] = []; 891 892 public totalCount(): number { 893 return this.dataArray.length; 894 } 895 896 public getData(index: number): string { 897 return this.dataArray[index]; 898 } 899 900 public addData(index: number, data: string): void { 901 this.dataArray.splice(index, 0, data); 902 this.notifyDataAdd(index); 903 } 904 905 public pushData(data: string): void { 906 this.dataArray.push(data); 907 this.notifyDataAdd(this.dataArray.length - 1); 908 } 909} 910 911@Reusable 912@Component({freezeWhenInactive: true}) 913struct ChildComponent { 914 @Link @Watch('descChange') desc: string; 915 @State item: string = ''; 916 @State index: number = 0; 917 descChange() { 918 console.info(`ChildComponent messageChange ${this.desc}`); 919 } 920 921 aboutToReuse(params: Record<string, ESObject>): void { 922 this.item = params.item; 923 this.index = params.index; 924 } 925 926 aboutToRecycle(): void { 927 console.info(`ChildComponent has been recycled`); 928 } 929 build() { 930 Column() { 931 Text(`ChildComponent index: ${this.index} item: ${this.item}`) 932 .fontSize(20) 933 Text(`desc: ${this.desc}`) 934 .fontSize(20) 935 }.border({width: 2, color: Color.Pink}) 936 } 937} 938 939@Entry 940@Component 941struct Page { 942 @State desc: string = 'Hello World'; 943 @State flag: boolean = true; 944 private data: MyDataSource = new MyDataSource(); 945 946 aboutToAppear() { 947 for (let i = 0; i < 50; i++) { 948 this.data.pushData(`Hello ${i}`); 949 } 950 } 951 952 build() { 953 Column() { 954 Button(`change desc`).onClick(() => { 955 hiTraceMeter.startTrace('change desc', 1); 956 this.desc += '!'; 957 hiTraceMeter.finishTrace('change desc', 1); 958 }) 959 960 Button(`change flag`).onClick(() => { 961 hiTraceMeter.startTrace('change flag', 1); 962 this.flag = !this.flag; 963 hiTraceMeter.finishTrace('change flag', 1); 964 }) 965 966 List({ space: 3 }) { 967 LazyForEach(this.data, (item: string, index: number) => { 968 ListItem() { 969 ChildComponent({index: index, item: item, desc: this.desc}).reuseId(index % 10 < 5 ? '1': '0') 970 } 971 }, (item: string) => item) 972 } 973 .cachedCount(5) 974 .height('60%') 975 976 if (this.flag) { 977 ChildComponent({index: -1, item: 'Hello', desc: this.desc}).reuseId( '1') 978 } 979 } 980 .height('100%') 981 } 982} 983``` 984 985### 组件混用 986 987组件冻结混用场景即当支持组件冻结的场景彼此之间组合使用,对于不同的API version版本,冻结行为会有不同。给父组件设置组件冻结标志,在API version 17及以下,当父组件解冻时,会解冻自己子组件所有的节点;从API version 18开始,父组件解冻时,只会解冻子组件的屏上节点。 988 989**Navigation和TabContent的混用** 990 991代码示例如下: 992 993```ts 994// index.ets 995@Component 996struct ChildOfParamComponent { 997 @Prop @Watch('onChange') child_val: number; 998 999 onChange() { 1000 console.info(`Appmonitor ChildOfParamComponent: child_val changed:${this.child_val}`); 1001 } 1002 1003 build() { 1004 Column() { 1005 Text(`Child Param: ${this.child_val}`); 1006 } 1007 } 1008} 1009 1010@Component 1011struct ParamComponent { 1012 @Prop @Watch('onChange') paramVal: number; 1013 1014 onChange() { 1015 console.info(`Appmonitor ParamComponent: paramVal changed:${this.paramVal}`); 1016 } 1017 1018 build() { 1019 Column() { 1020 Text(`val: ${this.paramVal}`) 1021 ChildOfParamComponent({ child_val: this.paramVal }); 1022 } 1023 } 1024} 1025 1026 1027 1028@Component 1029struct DelayComponent { 1030 @Prop @Watch('onChange') delayVal: number; 1031 1032 onChange() { 1033 console.info(`Appmonitor ParamComponent: delayVal changed:${this.delayVal}`); 1034 } 1035 1036 build() { 1037 Column() { 1038 Text(`Delay Param: ${this.delayVal}`); 1039 } 1040 } 1041} 1042 1043@Component({ freezeWhenInactive: true }) 1044struct TabsComponent { 1045 private controller: TabsController = new TabsController(); 1046 @State @Watch('onChange') tabState: number = 47; 1047 1048 onChange() { 1049 console.info(`Appmonitor TabsComponent: tabState changed:${this.tabState}`); 1050 } 1051 1052 build() { 1053 Column({ space: 10 }) { 1054 Button(`Incr state ${this.tabState}`) 1055 .fontSize(25) 1056 .onClick(() => { 1057 console.info('Button increment state value'); 1058 this.tabState = this.tabState + 1; 1059 }) 1060 1061 Tabs({ barPosition: BarPosition.Start, index: 0, controller: this.controller }) { 1062 TabContent() { 1063 ParamComponent({ paramVal: this.tabState }); 1064 }.tabBar('Update') 1065 1066 TabContent() { 1067 DelayComponent({ delayVal: this.tabState }); 1068 }.tabBar('DelayUpdate') 1069 } 1070 .vertical(false) 1071 .scrollable(true) 1072 .barMode(BarMode.Fixed) 1073 .barWidth(400) 1074 .barHeight(150) 1075 .animationDuration(400) 1076 .width('100%') 1077 .height(200) 1078 .backgroundColor(0xF5F5F5) 1079 } 1080 } 1081} 1082 1083@Entry 1084@Component 1085struct MyNavigationTestStack { 1086 @Provide('pageInfo') pageInfo: NavPathStack = new NavPathStack(); 1087 1088 @Builder 1089 PageMap(name: string) { 1090 if (name === 'pageOne') { 1091 PageOneStack() 1092 } else if (name === 'pageTwo') { 1093 PageTwoStack() 1094 } 1095 } 1096 1097 build() { 1098 Column() { 1099 Navigation(this.pageInfo) { 1100 Column() { 1101 Button('Next Page', { stateEffect: true, type: ButtonType.Capsule }) 1102 .width('80%') 1103 .height(40) 1104 .margin(20) 1105 .onClick(() => { 1106 this.pageInfo.pushPath({ name: 'pageOne' }); //将name指定的NavDestination页面信息入栈 1107 }) 1108 } 1109 }.title('NavIndex') 1110 .navDestination(this.PageMap) 1111 .mode(NavigationMode.Stack) 1112 } 1113 } 1114} 1115 1116@Component 1117struct PageOneStack { 1118 @Consume('pageInfo') pageInfo: NavPathStack; 1119 1120 build() { 1121 NavDestination() { 1122 Column() { 1123 TabsComponent(); 1124 1125 Button('Next Page', { stateEffect: true, type: ButtonType.Capsule }) 1126 .width('80%') 1127 .height(40) 1128 .margin(20) 1129 .onClick(() => { 1130 this.pageInfo.pushPathByName('pageTwo', null); 1131 }) 1132 }.width('100%').height('100%') 1133 }.title('pageOne') 1134 .onBackPressed(() => { 1135 this.pageInfo.pop(); 1136 return true; 1137 }) 1138 } 1139} 1140 1141@Component 1142struct PageTwoStack { 1143 @Consume('pageInfo') pageInfo: NavPathStack; 1144 1145 build() { 1146 NavDestination() { 1147 Column() { 1148 Button('Back Page', { stateEffect: true, type: ButtonType.Capsule }) 1149 .width('80%') 1150 .height(40) 1151 .margin(20) 1152 .onClick(() => { 1153 this.pageInfo.pop(); 1154 }) 1155 }.width('100%').height('100%') 1156 }.title('pageTwo') 1157 .onBackPressed(() => { 1158 this.pageInfo.pop(); 1159 return true; 1160 }) 1161 } 1162} 1163``` 1164 1165代码运行结果图如下: 1166 1167 1168 1169点击Button:Next Page,进入pageOne页面,页面中存在两个tab标签,默认在Update标签,开启组件冻结功能,Tabcontent的标签如果未被选中,状态变量不会刷新,如以下操作。 1170 1171点击Button:Incr state,日志中查询Appmonitor,存在3个打印。 1172 1173 1174 1175切换到DelayUpdate标签,点击Button:Incr state,日志中查询Appmonitor,存在2个打印。DelayUpdate中状态变量不会刷新与Update标签中相关的状态变量。 1176 1177 1178 1179在API version 17及以下: 1180 1181点击Next page进入下一个页面并返回,标签默认在DelayUpdate,再次点击Button:Incr state,日志中查询Appmonitor,存在4个打印,页面路由返回时,会解冻Tabcontent所有的标签。 1182 1183 1184 1185在API version 18及以上: 1186 1187点击Next page进入下一个页面并返回,标签默认在DelayUpdate,再次点击Button:Incr state,日志中查询Appmonitor,存在2个打印,页面路由返回时,只会解冻对应标签的节点。 1188 1189 1190 1191#### 页面和LazyForEach 1192 1193Navigation和TabContent混用时,之所以会解锁TabContent标签的子节点,是因为回到前一个页面时会从父组件开始递归解冻子组件,与此行为类似的还有页面生命周期:OnPageShow。OnPageShow会将当前Page中的根节点设置为active状态,TabContent作为页面的子节点,也会被设置为active状态。在屏幕灭屏和屏幕亮屏时会分别触发页面的生命周期:OnPageHide和OnPageShow,因此页面中使用LazyForEach时,手动灭屏和亮屏也能实现页面路由一样的效果,如以下示例代码: 1194 1195```ts 1196import { hiTraceMeter } from '@kit.PerformanceAnalysisKit'; 1197// 用于处理数据监听的IDataSource的基本实现 1198class BasicDataSource implements IDataSource { 1199 private listeners: DataChangeListener[] = []; 1200 private originDataArray: string[] = []; 1201 1202 public totalCount(): number { 1203 return 0; 1204 } 1205 1206 public getData(index: number): string { 1207 return this.originDataArray[index]; 1208 } 1209 1210 // 该方法为框架侧调用,为LazyForEach组件向其数据源处添加listener监听 1211 registerDataChangeListener(listener: DataChangeListener): void { 1212 if (this.listeners.indexOf(listener) < 0) { 1213 console.info('add listener'); 1214 this.listeners.push(listener); 1215 } 1216 } 1217 1218 // 该方法为框架侧调用,为对应的LazyForEach组件在数据源处去除listener监听 1219 unregisterDataChangeListener(listener: DataChangeListener): void { 1220 const pos = this.listeners.indexOf(listener); 1221 if (pos >= 0) { 1222 console.info('remove listener'); 1223 this.listeners.splice(pos, 1); 1224 } 1225 } 1226 1227 // 通知LazyForEach组件需要重载所有子组件 1228 notifyDataReload(): void { 1229 this.listeners.forEach(listener => { 1230 listener.onDataReloaded(); 1231 }) 1232 } 1233 1234 // 通知LazyForEach组件需要在index对应索引处添加子组件 1235 notifyDataAdd(index: number): void { 1236 this.listeners.forEach(listener => { 1237 listener.onDataAdd(index); 1238 }) 1239 } 1240 1241 // 通知LazyForEach组件在index对应索引处数据有变化,需要重建该子组件 1242 notifyDataChange(index: number): void { 1243 this.listeners.forEach(listener => { 1244 listener.onDataChange(index); 1245 }) 1246 } 1247 1248 // 通知LazyForEach组件需要在index对应索引处删除该子组件 1249 notifyDataDelete(index: number): void { 1250 this.listeners.forEach(listener => { 1251 listener.onDataDelete(index); 1252 }) 1253 } 1254 1255 // 通知LazyForEach组件将from索引和to索引处的子组件进行交换 1256 notifyDataMove(from: number, to: number): void { 1257 this.listeners.forEach(listener => { 1258 listener.onDataMove(from, to); 1259 }) 1260 } 1261} 1262 1263class MyDataSource extends BasicDataSource { 1264 private dataArray: string[] = []; 1265 1266 public totalCount(): number { 1267 return this.dataArray.length; 1268 } 1269 1270 public getData(index: number): string { 1271 return this.dataArray[index]; 1272 } 1273 1274 public addData(index: number, data: string): void { 1275 this.dataArray.splice(index, 0, data); 1276 this.notifyDataAdd(index); 1277 } 1278 1279 public pushData(data: string): void { 1280 this.dataArray.push(data); 1281 this.notifyDataAdd(this.dataArray.length - 1); 1282 } 1283} 1284 1285@Reusable 1286@Component({freezeWhenInactive: true}) 1287struct ChildComponent { 1288 @State desc: string = ''; 1289 @Link @Watch('sumChange') sum: number; 1290 1291 sumChange() { 1292 console.info(`sum: Change ${this.sum}`); 1293 } 1294 1295 aboutToReuse(params: Record<string, Object>): void { 1296 this.desc = params.desc as string; 1297 this.sum = params.sum as number; 1298 } 1299 1300 aboutToRecycle(): void { 1301 console.info(`ChildComponent has been recycled`); 1302 } 1303 build() { 1304 Column() { 1305 Divider() 1306 .color('#ff11acb8') 1307 Text(`子组件: ${this.desc}`) 1308 .fontSize(30) 1309 .fontWeight(30) 1310 Text(`${this.sum}`) 1311 .fontSize(30) 1312 .fontWeight(30) 1313 } 1314 } 1315} 1316 1317@Entry 1318@Component ({freezeWhenInactive: true}) 1319struct Page { 1320 private data: MyDataSource = new MyDataSource(); 1321 @State sum: number = 0; 1322 @State desc: string = ''; 1323 1324 aboutToAppear() { 1325 for (let index = 0; index < 20; index++) { 1326 this.data.pushData(index.toString()); 1327 } 1328 } 1329 1330 build() { 1331 Column() { 1332 Button(`add sum`).onClick(() => { 1333 this.sum++; 1334 }) 1335 .fontSize(30) 1336 .margin(20) 1337 List() { 1338 LazyForEach(this.data, (item: string) => { 1339 ListItem() { 1340 ChildComponent({desc: item, sum: this.sum}); 1341 } 1342 .width('100%') 1343 .height(100) 1344 }, (item: string) => item) 1345 }.cachedCount(5) 1346 } 1347 .height('100%') 1348 .width('100%') 1349 } 1350} 1351``` 1352 1353在组件复用场景中,已经对LazyForEach的节点进行了详细说明,分为屏上节点和cachedCount节点。 1354 1355 1356 1357向下滑动LazyForEach,让cachedCount补充节点,点击Button:add sum,搜索打印日志:sum:Change,出现了8条打印。 1358 1359 1360 1361在API version 17及以下: 1362 1363灭屏之后亮屏,触发OnPageShow,点击Button:add sum,打印数量 = 屏上节点 + cachedCount的数量。 1364 1365 1366 1367从API version 18开始: 1368 1369灭屏之后亮屏,触发OnPageShow,点击Button:add sum,只会打印屏上节点数量,不再会解冻cachedCount中的节点。 1370 1371 1372 1373## 限制条件 1374 1375如下面的例子所示,FreezeBuildNode中使用了自定义节点[BuilderNode](../../reference/apis-arkui/js-apis-arkui-builderNode.md)。BuilderNode可以通过命令式动态挂载组件,而组件冻结又是强依赖父子关系来通知是否开启组件冻结。如果父组件使用组件冻结,且组件树的中间层级上又启用了BuilderNode,则BuilderNode的子组件将无法被冻结。 1376 1377```ts 1378import { BuilderNode, FrameNode, NodeController, UIContext } from '@kit.ArkUI'; 1379 1380// 定义一个Params类,用于传递参数 1381class Params { 1382 index: number = 0; 1383 1384 constructor(index: number) { 1385 this.index = index; 1386 } 1387} 1388 1389// 定义一个BuildNodeChild组件,它包含一个message属性和一个index属性 1390@Component 1391struct BuildNodeChild { 1392 @StorageProp('buildNodeTest') @Watch('onMessageUpdated') message: string = 'hello world'; 1393 @State index: number = 0; 1394 1395 // 当message更新时,调用此方法 1396 onMessageUpdated() { 1397 console.info(`FreezeBuildNode builderNodeChild message callback func ${this.message},index:${this.index}`); 1398 } 1399 1400 build() { 1401 Text(`buildNode Child message: ${this.message}`).fontSize(30) 1402 } 1403} 1404 1405// 定义一个buildText函数,它接收一个Params参数并构建一个Column组件 1406@Builder 1407function buildText(params: Params) { 1408 Column() { 1409 BuildNodeChild({ index: params.index }) 1410 } 1411} 1412 1413// 定义一个TextNodeController类,继承自NodeController 1414class TextNodeController extends NodeController { 1415 private textNode: BuilderNode<[Params]> | null = null; 1416 private index: number = 0; 1417 1418 // 构造函数接收一个index参数 1419 constructor(index: number) { 1420 super(); 1421 this.index = index; 1422 } 1423 1424 // 创建并返回一个FrameNode 1425 makeNode(context: UIContext): FrameNode | null { 1426 this.textNode = new BuilderNode(context); 1427 this.textNode.build(wrapBuilder<[Params]>(buildText), new Params(this.index)); 1428 return this.textNode.getFrameNode(); 1429 } 1430} 1431 1432// 定义一个Index组件,它包含一个message属性和一个data数组 1433@Entry 1434@Component 1435struct Index { 1436 @StorageLink('buildNodeTest') message: string = 'hello'; 1437 private data: number[] = [0, 1]; 1438 1439 build() { 1440 Row() { 1441 Column() { 1442 Button('change').fontSize(30) 1443 .onClick(() => { 1444 this.message += 'a'; 1445 }) 1446 1447 Tabs() { 1448 ForEach(this.data, (item: number) => { 1449 TabContent() { 1450 FreezeBuildNode({ index: item }) 1451 }.tabBar(`tab${item}`) 1452 }, (item: number) => item.toString()) 1453 } 1454 } 1455 } 1456 .width('100%') 1457 .height('100%') 1458 } 1459} 1460 1461// 定义一个FreezeBuildNode组件,它包含一个message属性和一个index属性 1462@Component({ freezeWhenInactive: true }) 1463struct FreezeBuildNode { 1464 @StorageProp('buildNodeTest') @Watch('onMessageUpdated') message: string = '1111'; 1465 @State index: number = 0; 1466 1467 // 当message更新时,调用此方法 1468 onMessageUpdated() { 1469 console.info(`FreezeBuildNode message callback func ${this.message}, index: ${this.index}`); 1470 } 1471 1472 build() { 1473 NodeContainer(new TextNodeController(this.index)) 1474 .width('100%') 1475 .height('100%') 1476 .backgroundColor('#FFF0F0F0') 1477 } 1478} 1479``` 1480 1481在上面的示例中: 1482 1483点击Button('change')。改变message的值,当前正在显示的TabContent组件中的@Watch中注册的方法onMessageUpdated被触发。未显示的TabContent中的BuilderNode节点下组件的@Watch方法onMessageUpdated也被触发,并没有被冻结。 1484 1485 1486 1487