• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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![freezeInPage](./figures/freezeInPage.png)
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![freezeWithTab](./figures/freezewithTabs.png)
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![TabContent.gif](figures/TabContent.gif)
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![FrezzeLazyforEach.gif](figures/FrezzeLazyforEach.gif)
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![navigation-freeze.gif](figures/navigation-freeze.gif)
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![freeze](./figures/freezeResuable.png)
646可通过trace观察,仅触发了15个`ChildComponent`节点的刷新。
647![freeze](./figures/traceWithFreeze.png)
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![traceWithFreezeLazyForeachAndIf](./figures/traceWithFreezeLazyForeachAndIf.png)
815
816没有开启组件冻结trace:
817
818![traceWithFreezeLazyForeachAndIf](./figures/traceWithLazyForeachAndIf.png)
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![freeze](figures/freeze_tabcontent.gif)
1168
1169点击Button:Next Page,进入pageOne页面,页面中存在两个tab标签,默认在Update标签,开启组件冻结功能,Tabcontent的标签如果未被选中,状态变量不会刷新,如以下操作。
1170
1171点击Button:Incr state,日志中查询Appmonitor,存在3个打印。
1172
1173![freeze](figures/freeze_tabcontent_update.png)
1174
1175切换到DelayUpdate标签,点击Button:Incr state,日志中查询Appmonitor,存在2个打印。DelayUpdate中状态变量不会刷新与Update标签中相关的状态变量。
1176
1177![freeze](figures/freeze_tabcontent_delayupdate.png)
1178
1179在API version 17及以下:
1180
1181点击Next page进入下一个页面并返回,标签默认在DelayUpdate,再次点击Button:Incr state,日志中查询Appmonitor,存在4个打印,页面路由返回时,会解冻Tabcontent所有的标签。
1182
1183![freeze](figures/freeze_tabcontent_back_api15.png)
1184
1185在API version 18及以上:
1186
1187点击Next page进入下一个页面并返回,标签默认在DelayUpdate,再次点击Button:Incr state,日志中查询Appmonitor,存在2个打印,页面路由返回时,只会解冻对应标签的节点。
1188
1189![freeze](figures/freeze_tabcontent_back_api16.png)
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![freeze](figures/freeze_lazyforeach.png)
1356
1357向下滑动LazyForEach,让cachedCount补充节点,点击Button:add sum,搜索打印日志:sum:Change,出现了8条打印。
1358
1359![freeze](figures/freeze_lazyforeach_add.png)
1360
1361在API version 17及以下:
1362
1363灭屏之后亮屏,触发OnPageShow,点击Button:add sum,打印数量 = 屏上节点 + cachedCount的数量。
1364
1365![freeze](figures/freeze_lazyforeach_api15.png)
1366
1367从API version 18开始:
1368
1369灭屏之后亮屏,触发OnPageShow,点击Button:add sum,只会打印屏上节点数量,不再会解冻cachedCount中的节点。
1370
1371![freeze](figures/freeze_lazyforeach_api16.png)
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![builderNode.gif](figures/builderNode.gif)
1486
1487