• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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![freezeInPage](./figures/freezeInPage.png)
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![freezeWithTab](./figures/freezewithTabs.png)
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![TabContent.gif](figures/TabContent.gif)
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![FrezzeLazyforEach.gif](figures/FrezzeLazyforEach.gif)
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![navigation-freeze.gif](figures/navigation-freeze.gif)
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![freeze](./figures/freezeResuable.png)
635可通过trace观察,仅触发了15个`ChildComponent`节点的刷新。
636![freeze](./figures/traceWithFreeze.png)
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![traceWithFreezeLazyForeachAndIf](./figures/traceWithFreezeLazyForeachAndIf.png)
804
805没有开启组件冻结trace:
806
807![traceWithFreezeLazyForeachAndIf](./figures/traceWithLazyForeachAndIf.png)
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![freeze](figures/freeze_tabcontent.gif)
1155
1156点击Button:Next Page,进入pageOne页面,页面中存在两个tab标签,默认在Update标签,开启组件冻结功能,Tabcontent的标签如果未被选中,状态变量不会刷新,如以下操作。
1157
1158点击Button:Incr state,日志中查询Appmonitor,存在3个打印。
1159
1160![freeze](figures/freeze_tabcontent_update.png)
1161
1162切换到DelayUpdate标签,点击Button:Incr state,日志中查询Appmonitor,存在2个打印。DelayUpdate中状态变量不会刷新与Update标签中相关的状态变量。
1163
1164![freeze](figures/freeze_tabcontent_delayupdate.png)
1165
1166在API version 17及以下:
1167
1168点击Next page进入下一个页面并返回,标签默认在DelayUpdate,再次点击Button:Incr state,日志中查询Appmonitor,存在4个打印,页面路由返回时,会解冻Tabcontent所有的标签。
1169
1170![freeze](figures/freeze_tabcontent_back_api15.png)
1171
1172在API version 18及以上:
1173
1174点击Next page进入下一个页面并返回,标签默认在DelayUpdate,再次点击Button:Incr state,日志中查询Appmonitor,存在2个打印,页面路由返回时,只会解冻对应标签的节点。
1175
1176![freeze](figures/freeze_tabcontent_back_api16.png)
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![freeze](figures/freeze_lazyforeach.png)
1343
1344向下滑动LazyForEach,让cachedCount补充节点,点击Button:add sum,搜索打印日志:sum:Change,出现了8条打印。
1345
1346![freeze](figures/freeze_lazyforeach_add.png)
1347
1348在API version 17及以下:
1349
1350灭屏之后亮屏,触发OnPageShow,点击Button:add sum,打印数量 = 屏上节点 + cachedCount的数量。
1351
1352![freeze](figures/freeze_lazyforeach_api15.png)
1353
1354从API version 18开始:
1355
1356灭屏之后亮屏,触发OnPageShow,点击Button:add sum,只会打印屏上节点数量,不再会解冻cachedCount中的节点。
1357
1358![freeze](figures/freeze_lazyforeach_api16.png)
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![builderNode.gif](figures/builderNode.gif)
1473
1474