1# 自定义组件冻结功能 2 3自定义组件冻结功能专为优化复杂UI页面的性能而设计,尤其适用于包含多个页面栈、长列表或宫格布局的场景。在这些情况下,当状态变量绑定了多个UI组件,其变化可能触发大量UI组件的刷新,进而导致界面卡顿和响应延迟。为了提升这类负载UI界面的刷新性能,开发者可以选择尝试使用自定义组件冻结功能。 4 5组件冻结功能是一种性能优化机制,它会冻结非激活状态下的组件的刷新能力。当组件处于非激活状态时,即使其绑定的状态变量发生变化,也不会触发该组件的UI重新渲染,从而降低复杂UI场景下的刷新负载。 6 7组件冻结的工作原理是: 81. 开发者通过设置freezeWhenInactive属性,即可激活组件冻结机制。 92. 启用后,系统将仅对处于激活状态的自定义组件进行更新,这使得UI框架可以尽量缩小更新范围,仅限于用户可见范围内(激活状态)的自定义组件,从而提高复杂UI场景下的刷新效率。 103. 当之前处于inactive状态的自定义组件重新变为active状态时,状态管理框架会对其执行必要的刷新操作,确保UI的正确展示。 11 12简而言之,组件冻结旨在优化复杂界面下的UI刷新性能。在存在多个不可见自定义组件的情况下,如多页面栈、长列表或宫格,通过组件冻结可以实现按需刷新,即仅刷新当前可见的自定义组件,而将不可见自定义组件的刷新延迟至它们变为可见时。 13 14需要注意,组件active/inactive并不等同于其可见性。组件冻结目前仅适用于以下场景: 15 161. 页面路由:当前栈顶页面为active状态,非栈顶不可见页面为inactive状态。 172. TabContent:只有当前显示的TabContent中的自定义组件处于active状态,其余则为inactive。 183. LazyForEach:仅当前显示的LazyForEach中的自定义组件为active状态,而缓存节点的组件则为inactive状态。 194. Navigation:当前显示的NavDestination中的自定义组件为active状态,而其他未显示的NavDestination组件则为inactive状态。 205. 组件复用:进入复用池的组件为inactive状态,从复用池上树的节点为active状态。 216. 混用场景:对于以上场景的组合使用,例如TabContent下面使用LazyForEach,切换Tab时,API version 17及以下,LazyForEach中的所有节点都会被设置为active状态,而从API version 18开始,只有LazyForEach的屏上节点会被设置为active状态,其余则为inactive状态。 22 23在阅读本文档前,开发者需要了解自定义组件基本语法。建议提前阅读:[自定义组件](./arkts-create-custom-components.md)。 24 25> **说明:** 26> 27> 从API version 11开始,支持自定义组件冻结功能。 28> 29> 从API version 18开始,支持自定义组件冻结功能的混用场景冻结。 30 31## 当前支持的场景 32 33### 页面路由 34 35> **说明:** 36> 37> 本示例使用了router进行页面跳转,建议开发者使用组件导航(Navigation)代替页面路由(router)来实现页面切换。Navigation提供了更多的功能和更灵活的自定义能力。请参考[使用Navigation的组件冻结用例](#navigation)。 38 39当页面1调用router.pushUrl接口跳转到页面2时,页面1为隐藏不可见状态,此时如果更新页面1中的状态变量,不会触发页面1刷新。 40图示如下: 41 42 43 44页面1: 45 46```ts 47@Entry 48@Component({ freezeWhenInactive: true }) 49struct Page1 { 50 @StorageLink('PropA') @Watch("first") storageLink: number = 47; 51 52 first() { 53 console.info("first page " + `${this.storageLink}`) 54 } 55 56 build() { 57 Column() { 58 Text(`From first Page ${this.storageLink}`).fontSize(50) 59 Button('first page storageLink + 1').fontSize(30) 60 .onClick(() => { 61 this.storageLink += 1 62 }) 63 Button('go to next page').fontSize(30) 64 .onClick(() => { 65 this.getUIContext().getRouter().pushUrl({ url: 'pages/Page2' }); 66 }) 67 } 68 } 69} 70``` 71 72页面2: 73 74```ts 75@Entry 76@Component({ freezeWhenInactive: true }) 77struct Page2 { 78 @StorageLink('PropA') @Watch("second") storageLink2: number = 1; 79 80 second() { 81 console.info("second page: " + `${this.storageLink2}`) 82 } 83 84 build() { 85 Column() { 86 87 Text(`second Page ${this.storageLink2}`).fontSize(50) 88 Button('Change Divider.strokeWidth') 89 .onClick(() => { 90 this.getUIContext().getRouter().back(); 91 }) 92 93 Button('second page storageLink2 + 2').fontSize(30) 94 .onClick(() => { 95 this.storageLink2 += 2 96 }) 97 98 } 99 } 100} 101``` 102 103在上面的示例中: 104 1051.点击页面1中的Button “first page storageLink + 1”,storageLink状态变量改变,@Watch中注册的方法first会被调用。 106 1072.通过router.pushUrl({url: 'pages/second'}),跳转到页面2,页面1隐藏,状态由active变为inactive。 108 1093.点击页面2中的Button “this.storageLink2 += 2”,只回调页面2@Watch中注册的方法second,因为页面1的状态变量此时已被冻结。 110 1114.点击“back”,页面2被销毁,页面1的状态由inactive变为active,重新刷新在inactive时被冻结的状态变量,页面1@Watch中注册的方法first被再次调用。 112 113 114### TabContent 115 116- 对Tabs中当前不可见的TabContent进行冻结,不会触发组件的更新。 117 118- 需要注意的是:在首次渲染的时候,Tab只会创建当前正在显示的TabContent,当切换全部的TabContent后,TabContent才会被全部创建。 119 120图示如下: 121 122 123```ts 124@Entry 125@Component 126struct TabContentTest { 127 @State @Watch("onMessageUpdated") message: number = 0; 128 private data: number[] = [0, 1] 129 130 onMessageUpdated() { 131 console.info(`TabContent message callback func ${this.message}`) 132 } 133 134 build() { 135 Row() { 136 Column() { 137 Button('change message').onClick(() => { 138 this.message++ 139 }) 140 141 Tabs() { 142 ForEach(this.data, (item: number) => { 143 TabContent() { 144 FreezeChild({ message: this.message, index: item }) 145 }.tabBar(`tab${item}`) 146 }, (item: number) => item.toString()) 147 } 148 } 149 .width('100%') 150 } 151 .height('100%') 152 } 153} 154 155@Component({ freezeWhenInactive: true }) 156struct FreezeChild { 157 @Link @Watch("onMessageUpdated") message: number 158 private index: number = 0 159 160 onMessageUpdated() { 161 console.info(`FreezeChild message callback func ${this.message}, index: ${this.index}`) 162 } 163 164 build() { 165 Text("message" + `${this.message}, index: ${this.index}`) 166 .fontSize(50) 167 .fontWeight(FontWeight.Bold) 168 } 169} 170``` 171 172在上面的示例中: 173 1741.点击“change message”更改message的值,当前正在显示的TabContent组件中的@Watch中注册的方法onMessageUpdated被触发。 175 1762.点击“two”切换到另外的TabContent,TabContent状态由inactive变为active,对应的@Watch中注册的方法onMessageUpdated被触发。 177 1783.再次点击“change message”更改message的值,仅当前显示的TabContent子组件中的@Watch中注册的方法onMessageUpdated被触发。 179 180 181 182 183### LazyForEach 184 185- 对LazyForEach中缓存的自定义组件进行冻结,不会触发组件的更新。 186 187```ts 188// 用于处理数据监听的IDataSource的基本实现 189class BasicDataSource implements IDataSource { 190 private listeners: DataChangeListener[] = []; 191 private originDataArray: string[] = []; 192 193 public totalCount(): number { 194 return 0; 195 } 196 197 public getData(index: number): string { 198 return this.originDataArray[index]; 199 } 200 201 // 该方法为框架侧调用,为LazyForEach组件向其数据源处添加listener监听 202 registerDataChangeListener(listener: DataChangeListener): void { 203 if (this.listeners.indexOf(listener) < 0) { 204 console.info('add listener'); 205 this.listeners.push(listener); 206 } 207 } 208 209 // 该方法为框架侧调用,为对应的LazyForEach组件在数据源处去除listener监听 210 unregisterDataChangeListener(listener: DataChangeListener): void { 211 const pos = this.listeners.indexOf(listener); 212 if (pos >= 0) { 213 console.info('remove listener'); 214 this.listeners.splice(pos, 1); 215 } 216 } 217 218 // 通知LazyForEach组件需要重载所有子组件 219 notifyDataReload(): void { 220 this.listeners.forEach(listener => { 221 listener.onDataReloaded(); 222 }) 223 } 224 225 // 通知LazyForEach组件需要在index对应索引处添加子组件 226 notifyDataAdd(index: number): void { 227 this.listeners.forEach(listener => { 228 listener.onDataAdd(index); 229 }) 230 } 231 232 // 通知LazyForEach组件在index对应索引处数据有变化,需要重建该子组件 233 notifyDataChange(index: number): void { 234 this.listeners.forEach(listener => { 235 listener.onDataChange(index); 236 }) 237 } 238 239 // 通知LazyForEach组件需要在index对应索引处删除该子组件 240 notifyDataDelete(index: number): void { 241 this.listeners.forEach(listener => { 242 listener.onDataDelete(index); 243 }) 244 } 245} 246 247class MyDataSource extends BasicDataSource { 248 private dataArray: string[] = []; 249 250 public totalCount(): number { 251 return this.dataArray.length; 252 } 253 254 public getData(index: number): string { 255 return this.dataArray[index]; 256 } 257 258 public addData(index: number, data: string): void { 259 this.dataArray.splice(index, 0, data); 260 this.notifyDataAdd(index); 261 } 262 263 public pushData(data: string): void { 264 this.dataArray.push(data); 265 this.notifyDataAdd(this.dataArray.length - 1); 266 } 267} 268 269@Entry 270@Component 271struct LforEachTest { 272 private data: MyDataSource = new MyDataSource(); 273 @State @Watch("onMessageUpdated") message: number = 0; 274 275 onMessageUpdated() { 276 console.info(`LazyforEach message callback func ${this.message}`) 277 } 278 279 aboutToAppear() { 280 for (let i = 0; i <= 20; i++) { 281 this.data.pushData(`Hello ${i}`) 282 } 283 } 284 285 build() { 286 Column() { 287 Button('change message').onClick(() => { 288 this.message++ 289 }) 290 List({ space: 3 }) { 291 LazyForEach(this.data, (item: string) => { 292 ListItem() { 293 FreezeChild({ message: this.message, index: item }) 294 } 295 }, (item: string) => item) 296 }.cachedCount(5).height(500) 297 } 298 299 } 300} 301 302@Component({ freezeWhenInactive: true }) 303struct FreezeChild { 304 @Link @Watch("onMessageUpdated") message: number; 305 private index: string = ""; 306 307 aboutToAppear() { 308 console.info(`FreezeChild aboutToAppear index: ${this.index}`) 309 } 310 311 onMessageUpdated() { 312 console.info(`FreezeChild message callback func ${this.message}, index: ${this.index}`) 313 } 314 315 build() { 316 Text("message" + `${this.message}, index: ${this.index}`) 317 .width('90%') 318 .height(160) 319 .backgroundColor(0xAFEEEE) 320 .textAlign(TextAlign.Center) 321 .fontSize(30) 322 .fontWeight(FontWeight.Bold) 323 } 324} 325``` 326 327在上面的示例中: 328 3291.点击“change message”更改message的值,当前正在显示的ListItem中的子组件@Watch中注册的方法onMessageUpdated被触发。缓存节点@Watch中注册的方法不会被触发。(如果不加组件冻结,当前正在显示的ListItem和cachecount缓存节点@Watch中注册的方法onMessageUpdated都会触发watch回调。) 330 3312.List区域外的ListItem滑动到List区域内,状态由inactive变为active,对应的@Watch中注册的方法onMessageUpdated被触发。 332 3333.再次点击“change message”更改message的值,仅有当前显示的ListItem中的子组件@Watch中注册的方法onMessageUpdated被触发。 334 335 336 337### Navigation 338 339- 当NavDestination不可见时,会将其子自定义组件设置成非激活态,不会触发组件的刷新。当返回该页面时,其子自定义组件重新恢复成激活态,触发@Watch回调进行刷新。 340 341- 在下面例子中,NavigationContentMsgStack会被设置成非激活态,将不再响应状态变量的变化,也不会触发组件刷新。 342 343```ts 344@Entry 345@Component 346struct MyNavigationTestStack { 347 @Provide('pageInfo') pageInfo: NavPathStack = new NavPathStack(); 348 @State @Watch("info") message: number = 0; 349 @State logNumber: number = 0; 350 351 info() { 352 console.info(`freeze-test MyNavigation message callback ${this.message}`); 353 } 354 355 @Builder 356 PageMap(name: string) { 357 if (name === 'pageOne') { 358 pageOneStack({ message: this.message, logNumber: this.logNumber }) 359 } else if (name === 'pageTwo') { 360 pageTwoStack({ message: this.message, logNumber: this.logNumber }) 361 } else if (name === 'pageThree') { 362 pageThreeStack({ message: this.message, logNumber: this.logNumber }) 363 } 364 } 365 366 build() { 367 Column() { 368 Button('change message') 369 .onClick(() => { 370 this.message++; 371 }) 372 Navigation(this.pageInfo) { 373 Column() { 374 Button('Next Page', { stateEffect: true, type: ButtonType.Capsule }) 375 .width('80%') 376 .height(40) 377 .margin(20) 378 .onClick(() => { 379 this.pageInfo.pushPath({ name: 'pageOne' }); //将name指定的NavDestination页面信息入栈 380 }) 381 } 382 }.title('NavIndex') 383 .navDestination(this.PageMap) 384 .mode(NavigationMode.Stack) 385 } 386 } 387} 388 389@Component 390struct pageOneStack { 391 @Consume('pageInfo') pageInfo: NavPathStack; 392 @State index: number = 1; 393 @Link message: number; 394 @Link logNumber: number; 395 396 build() { 397 NavDestination() { 398 Column() { 399 NavigationContentMsgStack({ message: this.message, index: this.index, logNumber: this.logNumber }) 400 Text("cur stack size:" + `${this.pageInfo.size()}`) 401 .fontSize(30) 402 .fontWeight(FontWeight.Bold) 403 Button('Next Page', { stateEffect: true, type: ButtonType.Capsule }) 404 .width('80%') 405 .height(40) 406 .margin(20) 407 .onClick(() => { 408 this.pageInfo.pushPathByName('pageTwo', null); 409 }) 410 Button('Back Page', { stateEffect: true, type: ButtonType.Capsule }) 411 .width('80%') 412 .height(40) 413 .margin(20) 414 .onClick(() => { 415 this.pageInfo.pop(); 416 }) 417 }.width('100%').height('100%') 418 }.title('pageOne') 419 .onBackPressed(() => { 420 this.pageInfo.pop(); 421 return true; 422 }) 423 } 424} 425 426@Component 427struct pageTwoStack { 428 @Consume('pageInfo') pageInfo: NavPathStack; 429 @State index: number = 2; 430 @Link message: number; 431 @Link logNumber: number; 432 433 build() { 434 NavDestination() { 435 Column() { 436 NavigationContentMsgStack({ message: this.message, index: this.index, logNumber: this.logNumber }) 437 Text("cur stack size:" + `${this.pageInfo.size()}`) 438 .fontSize(30) 439 .fontWeight(FontWeight.Bold) 440 Button('Next Page', { stateEffect: true, type: ButtonType.Capsule }) 441 .width('80%') 442 .height(40) 443 .margin(20) 444 .onClick(() => { 445 this.pageInfo.pushPathByName('pageThree', null); 446 }) 447 Button('Back Page', { stateEffect: true, type: ButtonType.Capsule }) 448 .width('80%') 449 .height(40) 450 .margin(20) 451 .onClick(() => { 452 this.pageInfo.pop(); 453 }) 454 }.width('100%').height('100%') 455 }.title('pageTwo') 456 .onBackPressed(() => { 457 this.pageInfo.pop(); 458 return true; 459 }) 460 } 461} 462 463@Component 464struct pageThreeStack { 465 @Consume('pageInfo') pageInfo: NavPathStack; 466 @State index: number = 3; 467 @Link message: number; 468 @Link logNumber: number; 469 470 build() { 471 NavDestination() { 472 Column() { 473 NavigationContentMsgStack({ message: this.message, index: this.index, logNumber: this.logNumber }) 474 Text("cur stack size:" + `${this.pageInfo.size()}`) 475 .fontSize(30) 476 .fontWeight(FontWeight.Bold) 477 Button('Next Page', { stateEffect: true, type: ButtonType.Capsule }) 478 .width('80%') 479 .height(40) 480 .margin(20) 481 .onClick(() => { 482 this.pageInfo.pushPathByName('pageOne', null); 483 }) 484 Button('Back Page', { stateEffect: true, type: ButtonType.Capsule }) 485 .width('80%') 486 .height(40) 487 .margin(20) 488 .onClick(() => { 489 this.pageInfo.pop(); 490 }) 491 }.width('100%').height('100%') 492 }.title('pageThree') 493 .onBackPressed(() => { 494 this.pageInfo.pop(); 495 return true; 496 }) 497 } 498} 499 500@Component({ freezeWhenInactive: true }) 501struct NavigationContentMsgStack { 502 @Link @Watch("info") message: number; 503 @Link index: number; 504 @Link logNumber: number; 505 506 info() { 507 console.info(`freeze-test NavigationContent message callback ${this.message}`); 508 console.info(`freeze-test ---- called by content ${this.index}`); 509 this.logNumber++; 510 } 511 512 build() { 513 Column() { 514 Text("msg:" + `${this.message}`) 515 .fontSize(30) 516 .fontWeight(FontWeight.Bold) 517 Text("log number:" + `${this.logNumber}`) 518 .fontSize(30) 519 .fontWeight(FontWeight.Bold) 520 } 521 } 522} 523``` 524 525在上面的示例中: 526 5271.点击“change message”更改message的值,当前正在显示的MyNavigationTestStack组件中的@Watch中注册的方法info被触发。 528 5292.点击“Next Page”切换到PageOne,创建pageOneStack节点。 530 5313.再次点击“change message”更改message的值,仅pageOneStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。 532 5334.再次点击“Next Page”切换到PageTwo,创建pageTwoStack节点。 534 5355.再次点击“change message”更改message的值,仅pageTwoStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。 536 5376.再次点击“Next Page”切换到PageThree,创建pageThreeStack节点。 538 5397.再次点击“change message”更改message的值,仅pageThreeStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。 540 5418.点击“Back Page”回到PageTwo,此时,仅pageTwoStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。 542 5439.再次点击“Back Page”回到PageOne,此时,仅pageOneStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。 544 54510.再次点击“Back Page”回到初始页,此时,无任何触发。 546 547 548 549### 组件复用 550 551[组件复用](./arkts-reusable.md)通过重利用缓存池中已存在的节点,而非创建新节点,来优化UI性能并提升应用流畅度。复用池中的节点尽管未在UI组件树上展示,但是状态变量的更改仍会触发UI刷新。为了解决复用池中组件异常刷新问题,可以使用组件冻结避免复用池中的组件刷新。 552 553#### 组件复用、if和组件冻结混用场景 554下面是组件复用、if组件和组件冻结混合使用场景的例子,if组件绑定的状态变量变化成false时,触发子组件`ChildComponent`的下树,由于`ChildComponent`被标记了组件复用,所以不会被销毁,而是进入复用池,这个时候如果同时开启了组件冻结,则可以使在复用池里不再刷新。 555具体流程如下: 5561. 点击`change flag`,改变`flag`为false: 557 - 被标记\@Reusable的`ChildComponent`组件在下树时,不会被销毁,而是进入复用池,触发aboutToRecycle生命周期,同时设置状态为inactive。 558 - `ChildComponent`同时也开启了组件冻结,当其状态为inactive时,不会响应任何状态变量变化带来的UI刷新。 5592. 点击`change desc`,触发`Page`的成员变量`desc`的变化: 560 - `desc`是\@State装饰的,其变化会通知给其子组件`ChildComponent`\@Link装饰的`desc`。 561 - 但因为`ChildComponent`是inactive状态,且开启了组件冻结,所以这次变化并不会触发`@Watch('descChange')`的回调,以及`ChildComponent`UI刷新。如果没有开启组件冻结,当前`@Watch('descChange')`会立即回调,且复用池内的`ChildComponent`组件也会对应刷新。 5623. 再次点击`change flag`,改变`flag`为true: 563 - `ChildComponent`从复用池中重新加入到组件树上。 564 - 回调aboutToReuse生命周期,将当前最新的`count`值同步给子组件。`desc`是通过@State->@Link同步的,所以无需开发者手动在aboutToReuse中赋值。 565 - 设置ChildComponent为active状态,并且刷新在inactive时没有刷新的组件,在当前例子中,就是Text(ChildComponent desc: ${this.desc})。 566 567 568```ts 569@Reusable 570@Component({freezeWhenInactive: true}) 571struct ChildComponent { 572 @Link @Watch('descChange') desc: string; 573 @State count: number = 0; 574 descChange() { 575 console.info(`ChildComponent messageChange ${this.desc}`); 576 } 577 578 aboutToReuse(params: Record<string, ESObject>): void { 579 this.count = params.count as number; 580 } 581 582 aboutToRecycle(): void { 583 console.info(`ChildComponent has been recycled`); 584 } 585 build() { 586 Column() { 587 Text(`ChildComponent desc: ${this.desc}`) 588 .fontSize(20) 589 Text(`ChildComponent count ${this.count}`) 590 .fontSize(20) 591 }.border({width: 2, color: Color.Pink}) 592 } 593} 594 595@Entry 596@Component 597struct Page { 598 @State desc: string = 'Hello World'; 599 @State flag: boolean = true; 600 @State count: number = 0; 601 build() { 602 Column() { 603 Button(`change desc`).onClick(() => { 604 this.desc += '!'; 605 }) 606 Button(`change flag`).onClick(() => { 607 this.count++; 608 this.flag =! this.flag; 609 }) 610 if (this.flag) { 611 ChildComponent({desc: this.desc, count: this.count}) 612 } 613 } 614 .height('100%') 615 } 616} 617``` 618#### LazyForEach、组件复用和组件冻结混用场景 619在数据很多的长列表滑动场景下,开发者会使用LazyForEach来按需创建组件,同时配合组件复用降低在滑动过程中因创建和销毁组件带来的开销。 620但是开发者如果根据其复用类型不同,设置了<!--RP2-->[reuseId](../../performance/component-recycle.md#接口说明)<!--RP2End-->,或者为了保证滑动性能设置了较大的cacheCount,这就可能使复用池或者LazyForEach缓存较多的节点。 621在这种情况下,如果开发者触发List下所有子节点的刷新,就会带来节点刷新数量过大的问题,这个时候,可以考虑搭配组件冻结使用。 622 623如下面例子: 6241. 滑动到index为14的位置,当前屏幕上可见区域内有15个`ChildComponent`。 6252. 在滑动过程中: 626 - 列表上端的`ChildComponent`滑出可视区域外,此时先进入LazyForEach的缓存区域内,被设置inactive。在滑出LazyForEach缓存区域外后,因为标记了组件复用,所以并不会被析构,而是会进入复用池,此时再次被设置inactive。 627 - 列表下端LazyForEach的缓存节点会进入List范围内,此时会试图请求创建新的节点进入LazyForEach的缓存,发现有可复用的节点时,从复用池中拿出已有节点,触发aboutToReuse生命周期回调,此时因为节点进入的是LazyForEach的缓存区域,所以其状态依旧是inactive。 6283. 点击`change desc`,触发`Page`的成员变量`desc`的变化: 629 - `desc`是\@State装饰的,其变化会通知给其子组件`ChildComponent`\@Link装饰的`desc`。 630 - 非可视区域内的`ChildComponent`是inactive状态,且开启了组件冻结,所以这次变化只触发可视区域内的15个节点的`@Watch('descChange')`回调,并只刷新对应可视区域内的15个节点。LazyForEach和复用池中的节点并不会刷新,也不会触发\@Watch回调。 631 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 976组件冻结混用场景即当支持组件冻结的场景彼此之间组合使用,对于不同的API version版本,冻结行为会有不同。给父组件设置组件冻结标志,在API version 17及以下,当父组件解冻时,会解冻自己子组件所有的节点;从API version 18开始,父组件解冻时,只会解冻子组件的屏上节点。 977 978#### Navigation和TabContent的混用 979 980代码示例如下: 981 982```ts 983// index.ets 984@Component 985struct ChildOfParamComponent { 986 @Prop @Watch('onChange') child_val: number; 987 988 onChange() { 989 console.log(`Appmonitor ChildOfParamComponent: child_val changed:${this.child_val}`); 990 } 991 992 build() { 993 Column() { 994 Text(`Child Param: ${this.child_val}`); 995 } 996 } 997} 998 999@Component 1000struct ParamComponent { 1001 @Prop @Watch('onChange') paramVal: number; 1002 1003 onChange() { 1004 console.log(`Appmonitor ParamComponent: paramVal changed:${this.paramVal}`); 1005 } 1006 1007 build() { 1008 Column() { 1009 Text(`val: ${this.paramVal}`) 1010 ChildOfParamComponent({child_val: this.paramVal}); 1011 } 1012 } 1013} 1014 1015 1016 1017@Component 1018struct DelayComponent { 1019 @Prop @Watch('onChange') delayVal: number; 1020 1021 onChange() { 1022 console.log(`Appmonitor ParamComponent: delayVal changed:${this.delayVal}`); 1023 } 1024 1025 1026 build() { 1027 Column() { 1028 Text(`Delay Param: ${this.delayVal}`); 1029 } 1030 } 1031} 1032 1033@Component ({freezeWhenInactive: true}) 1034struct TabsComponent { 1035 private controller: TabsController = new TabsController(); 1036 @State @Watch('onChange') tabState: number = 47; 1037 1038 onChange() { 1039 console.log(`Appmonitor TabsComponent: tabState changed:${this.tabState}`); 1040 } 1041 1042 build() { 1043 Column({space: 10}) { 1044 Button(`Incr state ${this.tabState}`) 1045 .fontSize(25) 1046 .onClick(() => { 1047 console.log('Button increment state value'); 1048 this.tabState = this.tabState + 1; 1049 }) 1050 1051 Tabs({ barPosition: BarPosition.Start, index: 0, controller: this.controller}) { 1052 TabContent() { 1053 ParamComponent({paramVal: this.tabState}); 1054 }.tabBar('Update') 1055 TabContent() { 1056 DelayComponent({delayVal: this.tabState}); 1057 }.tabBar('DelayUpdate') 1058 } 1059 .vertical(false) 1060 .scrollable(true) 1061 .barMode(BarMode.Fixed) 1062 .barWidth(400).barHeight(150).animationDuration(400) 1063 .width('100%') 1064 .height(200) 1065 .backgroundColor(0xF5F5F5) 1066 } 1067 } 1068} 1069 1070@Entry 1071@Component 1072struct MyNavigationTestStack { 1073 @Provide('pageInfo') pageInfo: NavPathStack = new NavPathStack(); 1074 1075 @Builder 1076 PageMap(name: string) { 1077 if (name === 'pageOne') { 1078 pageOneStack() 1079 } else if (name === 'pageTwo') { 1080 pageTwoStack() 1081 } 1082 } 1083 1084 build() { 1085 Column() { 1086 Navigation(this.pageInfo) { 1087 Column() { 1088 Button('Next Page', { stateEffect: true, type: ButtonType.Capsule }) 1089 .width('80%') 1090 .height(40) 1091 .margin(20) 1092 .onClick(() => { 1093 this.pageInfo.pushPath({ name: 'pageOne' }); //将name指定的NavDestination页面信息入栈 1094 }) 1095 } 1096 }.title('NavIndex') 1097 .navDestination(this.PageMap) 1098 .mode(NavigationMode.Stack) 1099 } 1100 } 1101} 1102 1103@Component 1104struct pageOneStack { 1105 @Consume('pageInfo') pageInfo: NavPathStack; 1106 1107 build() { 1108 NavDestination() { 1109 Column() { 1110 TabsComponent(); 1111 1112 Button('Next Page', { stateEffect: true, type: ButtonType.Capsule }) 1113 .width('80%') 1114 .height(40) 1115 .margin(20) 1116 .onClick(() => { 1117 this.pageInfo.pushPathByName('pageTwo', null); 1118 }) 1119 }.width('100%').height('100%') 1120 }.title('pageOne') 1121 .onBackPressed(() => { 1122 this.pageInfo.pop(); 1123 return true; 1124 }) 1125 } 1126} 1127 1128@Component 1129struct pageTwoStack { 1130 @Consume('pageInfo') pageInfo: NavPathStack; 1131 1132 build() { 1133 NavDestination() { 1134 Column() { 1135 Button('Back Page', { stateEffect: true, type: ButtonType.Capsule }) 1136 .width('80%') 1137 .height(40) 1138 .margin(20) 1139 .onClick(() => { 1140 this.pageInfo.pop(); 1141 }) 1142 }.width('100%').height('100%') 1143 }.title('pageTwo') 1144 .onBackPressed(() => { 1145 this.pageInfo.pop(); 1146 return true; 1147 }) 1148 } 1149} 1150``` 1151 1152代码运行结果图如下: 1153 1154 1155 1156点击Button:Next Page,进入pageOne页面,页面中存在两个tab标签,默认在Update标签,开启组件冻结功能,Tabcontent的标签如果未被选中,状态变量不会刷新,如以下操作。 1157 1158点击Button:Incr state,日志中查询Appmonitor,存在3个打印。 1159 1160 1161 1162切换到DelayUpdate标签,点击Button:Incr state,日志中查询Appmonitor,存在2个打印。DelayUpdate中状态变量不会刷新与Update标签中相关的状态变量。 1163 1164 1165 1166在API version 17及以下: 1167 1168点击Next page进入下一个页面并返回,标签默认在DelayUpdate,再次点击Button:Incr state,日志中查询Appmonitor,存在4个打印,页面路由返回时,会解冻Tabcontent所有的标签。 1169 1170 1171 1172在API version 18及以上: 1173 1174点击Next page进入下一个页面并返回,标签默认在DelayUpdate,再次点击Button:Incr state,日志中查询Appmonitor,存在2个打印,页面路由返回时,只会解冻对应标签的节点。 1175 1176 1177 1178#### 页面和LazyForEach 1179 1180Navigation和TabContent混用时,之所以会解锁TabContent标签的子节点,是因为回到前一个页面时会从父组件开始递归解冻子组件,与此行为类似的还有页面生命周期:OnPageShow。OnPageShow会将当前Page中的根节点设置为active状态,TabContent作为页面的子节点,也会被设置为active状态。在屏幕灭屏和屏幕亮屏时会分别触发页面的生命周期:OnPageHide和OnPageShow,因此页面中使用LazyForEach时,手动灭屏和亮屏也能实现页面路由一样的效果,如以下示例代码: 1181 1182```ts 1183import { hiTraceMeter } from '@kit.PerformanceAnalysisKit'; 1184// 用于处理数据监听的IDataSource的基本实现 1185class BasicDataSource implements IDataSource { 1186 private listeners: DataChangeListener[] = []; 1187 private originDataArray: string[] = []; 1188 1189 public totalCount(): number { 1190 return 0; 1191 } 1192 1193 public getData(index: number): string { 1194 return this.originDataArray[index]; 1195 } 1196 1197 // 该方法为框架侧调用,为LazyForEach组件向其数据源处添加listener监听 1198 registerDataChangeListener(listener: DataChangeListener): void { 1199 if (this.listeners.indexOf(listener) < 0) { 1200 console.info('add listener'); 1201 this.listeners.push(listener); 1202 } 1203 } 1204 1205 // 该方法为框架侧调用,为对应的LazyForEach组件在数据源处去除listener监听 1206 unregisterDataChangeListener(listener: DataChangeListener): void { 1207 const pos = this.listeners.indexOf(listener); 1208 if (pos >= 0) { 1209 console.info('remove listener'); 1210 this.listeners.splice(pos, 1); 1211 } 1212 } 1213 1214 // 通知LazyForEach组件需要重载所有子组件 1215 notifyDataReload(): void { 1216 this.listeners.forEach(listener => { 1217 listener.onDataReloaded(); 1218 }) 1219 } 1220 1221 // 通知LazyForEach组件需要在index对应索引处添加子组件 1222 notifyDataAdd(index: number): void { 1223 this.listeners.forEach(listener => { 1224 listener.onDataAdd(index); 1225 }) 1226 } 1227 1228 // 通知LazyForEach组件在index对应索引处数据有变化,需要重建该子组件 1229 notifyDataChange(index: number): void { 1230 this.listeners.forEach(listener => { 1231 listener.onDataChange(index); 1232 }) 1233 } 1234 1235 // 通知LazyForEach组件需要在index对应索引处删除该子组件 1236 notifyDataDelete(index: number): void { 1237 this.listeners.forEach(listener => { 1238 listener.onDataDelete(index); 1239 }) 1240 } 1241 1242 // 通知LazyForEach组件将from索引和to索引处的子组件进行交换 1243 notifyDataMove(from: number, to: number): void { 1244 this.listeners.forEach(listener => { 1245 listener.onDataMove(from, to); 1246 }) 1247 } 1248} 1249 1250class MyDataSource extends BasicDataSource { 1251 private dataArray: string[] = []; 1252 1253 public totalCount(): number { 1254 return this.dataArray.length; 1255 } 1256 1257 public getData(index: number): string { 1258 return this.dataArray[index]; 1259 } 1260 1261 public addData(index: number, data: string): void { 1262 this.dataArray.splice(index, 0, data); 1263 this.notifyDataAdd(index); 1264 } 1265 1266 public pushData(data: string): void { 1267 this.dataArray.push(data); 1268 this.notifyDataAdd(this.dataArray.length - 1); 1269 } 1270} 1271 1272@Reusable 1273@Component({freezeWhenInactive: true}) 1274struct ChildComponent { 1275 @State desc: string = ''; 1276 @Link @Watch('sumChange') sum: number; 1277 1278 sumChange() { 1279 console.info(`sum: Change ${this.sum}`); 1280 } 1281 1282 aboutToReuse(params: Record<string, Object>): void { 1283 this.desc = params.desc as string; 1284 this.sum = params.sum as number; 1285 } 1286 1287 aboutToRecycle(): void { 1288 console.info(`ChildComponent has been recycled`); 1289 } 1290 build() { 1291 Column() { 1292 Divider() 1293 .color('#ff11acb8') 1294 Text(`子组件: ${this.desc}`) 1295 .fontSize(30) 1296 .fontWeight(30) 1297 Text(`${this.sum}`) 1298 .fontSize(30) 1299 .fontWeight(30) 1300 } 1301 } 1302} 1303 1304@Entry 1305@Component ({freezeWhenInactive: true}) 1306struct Page { 1307 private data: MyDataSource = new MyDataSource(); 1308 @State sum: number = 0; 1309 @State desc: string = ''; 1310 1311 aboutToAppear() { 1312 for (let index = 0; index < 20; index++) { 1313 this.data.pushData(index.toString()); 1314 } 1315 } 1316 1317 build() { 1318 Column() { 1319 Button(`add sum`).onClick(() => { 1320 this.sum++; 1321 }) 1322 .fontSize(30) 1323 .margin(20) 1324 List() { 1325 LazyForEach(this.data, (item: string) => { 1326 ListItem() { 1327 ChildComponent({desc: item, sum: this.sum}); 1328 } 1329 .width('100%') 1330 .height(100) 1331 }, (item: string) => item) 1332 }.cachedCount(5) 1333 } 1334 .height('100%') 1335 .width('100%') 1336 } 1337} 1338``` 1339 1340在组件复用场景中,已经对LazyForEach的节点进行了详细说明,分为屏上节点和cachedCount节点。 1341 1342 1343 1344向下滑动LazyForEach,让cachedCount补充节点,点击Button:add sum,搜索打印日志:sum:Change,出现了8条打印。 1345 1346 1347 1348在API version 17及以下: 1349 1350灭屏之后亮屏,触发OnPageShow,点击Button:add sum,打印数量 = 屏上节点 + cachedCount的数量。 1351 1352 1353 1354从API version 18开始: 1355 1356灭屏之后亮屏,触发OnPageShow,点击Button:add sum,只会打印屏上节点数量,不再会解冻cachedCount中的节点。 1357 1358 1359 1360## 限制条件 1361 1362如下面的例子所示,FreezeBuildNode中使用了自定义节点[BuilderNode](../../reference/apis-arkui/js-apis-arkui-builderNode.md)。BuilderNode可以通过命令式动态挂载组件,而组件冻结又是强依赖父子关系来通知是否开启组件冻结。如果父组件使用组件冻结,且组件树的中间层级上又启用了BuilderNode,则BuilderNode的子组件将无法被冻结。 1363 1364``` 1365import { BuilderNode, FrameNode, NodeController, UIContext } from '@kit.ArkUI'; 1366 1367// 定义一个Params类,用于传递参数 1368class Params { 1369 index: number = 0; 1370 1371 constructor(index: number) { 1372 this.index = index; 1373 } 1374} 1375 1376// 定义一个buildNodeChild组件,它包含一个message属性和一个index属性 1377@Component 1378struct buildNodeChild { 1379 @StorageProp("buildNodeTest") @Watch("onMessageUpdated") message: string = "hello world"; 1380 @State index: number = 0; 1381 1382 // 当message更新时,调用此方法 1383 onMessageUpdated() { 1384 console.log(`FreezeBuildNode builderNodeChild message callback func ${this.message},index:${this.index}`); 1385 } 1386 1387 build() { 1388 Text(`buildNode Child message: ${this.message}`).fontSize(30) 1389 } 1390} 1391 1392// 定义一个buildText函数,它接收一个Params参数并构建一个Column组件 1393@Builder 1394function buildText(params: Params) { 1395 Column() { 1396 buildNodeChild({ index: params.index }) 1397 } 1398} 1399 1400// 定义一个TextNodeController类,继承自NodeController 1401class TextNodeController extends NodeController { 1402 private textNode: BuilderNode<[Params]> | null = null; 1403 private index: number = 0; 1404 1405 // 构造函数接收一个index参数 1406 constructor(index: number) { 1407 super(); 1408 this.index = index; 1409 } 1410 1411 // 创建并返回一个FrameNode 1412 makeNode(context: UIContext): FrameNode | null { 1413 this.textNode = new BuilderNode(context); 1414 this.textNode.build(wrapBuilder<[Params]>(buildText), new Params(this.index)); 1415 return this.textNode.getFrameNode(); 1416 } 1417} 1418 1419// 定义一个Index组件,它包含一个message属性和一个data数组 1420@Entry 1421@Component 1422struct Index { 1423 @StorageLink("buildNodeTest") message: string = "hello"; 1424 private data: number[] = [0, 1]; 1425 1426 build() { 1427 Row() { 1428 Column() { 1429 Button("change").fontSize(30) 1430 .onClick(() => { 1431 this.message += 'a'; 1432 }) 1433 1434 Tabs() { 1435 ForEach(this.data, (item: number) => { 1436 TabContent() { 1437 FreezeBuildNode({ index: item }) 1438 }.tabBar(`tab${item}`) 1439 }, (item: number) => item.toString()) 1440 } 1441 } 1442 } 1443 .width('100%') 1444 .height('100%') 1445 } 1446} 1447 1448// 定义一个FreezeBuildNode组件,它包含一个message属性和一个index属性 1449@Component({ freezeWhenInactive: true }) 1450struct FreezeBuildNode { 1451 @StorageProp("buildNodeTest") @Watch("onMessageUpdated") message: string = "1111"; 1452 @State index: number = 0; 1453 1454 // 当message更新时,调用此方法 1455 onMessageUpdated() { 1456 console.log(`FreezeBuildNode message callback func ${this.message}, index: ${this.index}`); 1457 } 1458 1459 build() { 1460 NodeContainer(new TextNodeController(this.index)) 1461 .width('100%') 1462 .height('100%') 1463 .backgroundColor('#FFF0F0F0') 1464 } 1465} 1466``` 1467 1468在上面的示例中: 1469 1470点击Button("change")。改变message的值,当前正在显示的TabContent组件中的@Watch中注册的方法onMessageUpdated被触发。未显示的TabContent中的BuilderNode节点下组件的@Watch方法onMessageUpdated也被触发,并没有被冻结。 1471 1472 1473 1474