1# 自定义组件冻结功能
2
3自定义组件冻结功能专为优化复杂UI页面的性能而设计,尤其适用于包含多个页面栈、长列表或宫格布局的场景。在这些情况下,当状态变量绑定了多个UI组件,其变化可能触发大量UI组件的刷新,进而导致界面卡顿和响应延迟。为了提升这类负载UI界面的刷新性能,开发者可以选择尝试使用自定义组件冻结功能。
4
5组件冻结的工作原理是:
61. 开发者通过设置freezeWhenInactive属性,即可激活组件冻结机制。
72. 启用后,系统将仅对处于激活状态的自定义组件进行更新,这使得UI框架可以尽量缩小更新范围,仅限于用户可见范围内(激活状态)的自定义组件,从而提高复杂UI场景下的刷新效率。
83. 当之前处于inactive状态的自定义组件重新变为active状态时,状态管理框架会对其执行必要的刷新操作,确保UI的正确展示。
9
10简而言之,组件冻结旨在优化复杂界面下的UI刷新性能。在存在多个不可见自定义组件的情况下,如多页面栈、长列表或宫格,通过组件冻结可以实现按需刷新,即仅刷新当前可见的自定义组件,而将不可见自定义组件的刷新延迟至它们变为可见时。
11
12需要注意,组件active/inactive并不等同于其可见性。组件冻结目前仅适用于以下场景:
13
141. 页面路由:当前栈顶页面为active状态,非栈顶不可见页面为inactive状态。
152. TabContent:只有当前显示的TabContent中的自定义组件处于active状态,其余则为inactive。
163. LazyForEach:仅当前显示的LazyForEach中的自定义组件为active状态,而缓存节点的组件则为inactive状态。
174. Navigation:当前显示的NavDestination中的自定义组件为active状态,而其他未显示的NavDestination组件则为inactive状态。
185. 组件复用:进入复用池的组件为inactive状态,从复用池上树的节点为active状态。
19其他场景,如堆叠布局(Stack)下的被遮罩的组件,这些组件尽管不可见,但并不被视为inactive状态,因此不在组件冻结的适用范围内。
20
21
22在阅读本文档前,开发者需要了解自定义组件基本语法。建议提前阅读:[自定义组件](./arkts-create-custom-components.md)。
23
24> **说明:**
25>
26> 从API version 11开始,支持自定义组件冻结功能。
27
28## 当前支持的场景
29
30### 页面路由
31
32> **说明:**
33>
34> 本示例使用了router进行页面跳转,建议开发者使用组件导航(Navigation)代替页面路由(router)来实现页面切换。Navigation提供了更多的功能和更灵活的自定义能力。请参考[使用Navigation的组件冻结用例](#navigation)。
35
36当页面1调用router.pushUrl接口跳转到页面2时,页面1为隐藏不可见状态,此时如果更新页面1中的状态变量,不会触发页面1刷新。
37图示如下:
38
39![freezeInPage](./figures/freezeInPage.png)
40
41页面1:
42
43```ts
44import { router } from '@kit.ArkUI';
45
46@Entry
47@Component({ freezeWhenInactive: true })
48struct Page1 {
49  @StorageLink('PropA') @Watch("first") storageLink: number = 47;
50
51  first() {
52    console.info("first page " + `${this.storageLink}`)
53  }
54
55  build() {
56    Column() {
57      Text(`From first Page ${this.storageLink}`).fontSize(50)
58      Button('first page storageLink + 1').fontSize(30)
59        .onClick(() => {
60          this.storageLink += 1
61        })
62      Button('go to next page').fontSize(30)
63        .onClick(() => {
64          router.pushUrl({ url: 'pages/Page2' })
65        })
66    }
67  }
68}
69```
70
71页面2:
72
73```ts
74import { router } from '@kit.ArkUI';
75
76@Entry
77@Component({ freezeWhenInactive: true })
78struct Page2 {
79  @StorageLink('PropA') @Watch("second") storageLink2: number = 1;
80
81  second() {
82    console.info("second page: " + `${this.storageLink2}`)
83  }
84
85  build() {
86    Column() {
87
88      Text(`second Page ${this.storageLink2}`).fontSize(50)
89      Button('Change Divider.strokeWidth')
90        .onClick(() => {
91          router.back()
92        })
93
94      Button('second page storageLink2 + 2').fontSize(30)
95        .onClick(() => {
96          this.storageLink2 += 2
97        })
98
99    }
100  }
101}
102```
103
104在上面的示例中:
105
1061.点击页面1中的Button “first page storageLink + 1”,storageLink状态变量改变,@Watch中注册的方法first会被调用。
107
1082.通过router.pushUrl({url: 'pages/second'}),跳转到页面2,页面1隐藏,状态由active变为inactive。
109
1103.点击页面2中的Button “this.storageLink2 += 2”,只回调页面2@Watch中注册的方法second,因为页面1的状态变量此时已被冻结。
111
1124.点击“back”,页面2被销毁,页面1的状态由inactive变为active,重新刷新在inactive时被冻结的状态变量,页面1@Watch中注册的方法first被再次调用。
113
114
115### TabContent
116
117- 对Tabs中当前不可见的TabContent进行冻结,不会触发组件的更新。
118
119- 需要注意的是:在首次渲染的时候,Tab只会创建当前正在显示的TabContent,当切换全部的TabContent后,TabContent才会被全部创建。
120
121图示如下:
122![freezeWithTab](./figures/freezewithTabs.png)
123
124```ts
125@Entry
126@Component
127struct TabContentTest {
128  @State @Watch("onMessageUpdated") message: number = 0;
129  private data: number[] = [0, 1]
130
131  onMessageUpdated() {
132    console.info(`TabContent message callback func ${this.message}`)
133  }
134
135  build() {
136    Row() {
137      Column() {
138        Button('change message').onClick(() => {
139          this.message++
140        })
141
142        Tabs() {
143          ForEach(this.data, (item: number) => {
144            TabContent() {
145              FreezeChild({ message: this.message, index: item })
146            }.tabBar(`tab${item}`)
147          }, (item: number) => item.toString())
148        }
149      }
150      .width('100%')
151    }
152    .height('100%')
153  }
154}
155
156@Component({ freezeWhenInactive: true })
157struct FreezeChild {
158  @Link @Watch("onMessageUpdated") message: number
159  private index: number = 0
160
161  onMessageUpdated() {
162    console.info(`FreezeChild message callback func ${this.message}, index: ${this.index}`)
163  }
164
165  build() {
166    Text("message" + `${this.message}, index: ${this.index}`)
167      .fontSize(50)
168      .fontWeight(FontWeight.Bold)
169  }
170}
171```
172
173在上面的示例中:
174
1751.点击“change message”更改message的值,当前正在显示的TabContent组件中的@Watch中注册的方法onMessageUpdated被触发。
176
1772.点击“two”切换到另外的TabContent,TabContent状态由inactive变为active,对应的@Watch中注册的方法onMessageUpdated被触发。
178
1793.再次点击“change message”更改message的值,仅当前显示的TabContent子组件中的@Watch中注册的方法onMessageUpdated被触发。
180
181![TabContent.gif](figures/TabContent.gif)
182
183
184### LazyForEach
185
186- 对LazyForEach中缓存的自定义组件进行冻结,不会触发组件的更新。
187
188```ts
189// 用于处理数据监听的IDataSource的基本实现
190class BasicDataSource implements IDataSource {
191  private listeners: DataChangeListener[] = [];
192  private originDataArray: string[] = [];
193
194  public totalCount(): number {
195    return 0;
196  }
197
198  public getData(index: number): string {
199    return this.originDataArray[index];
200  }
201
202  // 该方法为框架侧调用,为LazyForEach组件向其数据源处添加listener监听
203  registerDataChangeListener(listener: DataChangeListener): void {
204    if (this.listeners.indexOf(listener) < 0) {
205      console.info('add listener');
206      this.listeners.push(listener);
207    }
208  }
209
210  // 该方法为框架侧调用,为对应的LazyForEach组件在数据源处去除listener监听
211  unregisterDataChangeListener(listener: DataChangeListener): void {
212    const pos = this.listeners.indexOf(listener);
213    if (pos >= 0) {
214      console.info('remove listener');
215      this.listeners.splice(pos, 1);
216    }
217  }
218
219  // 通知LazyForEach组件需要重载所有子组件
220  notifyDataReload(): void {
221    this.listeners.forEach(listener => {
222      listener.onDataReloaded();
223    })
224  }
225
226  // 通知LazyForEach组件需要在index对应索引处添加子组件
227  notifyDataAdd(index: number): void {
228    this.listeners.forEach(listener => {
229      listener.onDataAdd(index);
230    })
231  }
232
233  // 通知LazyForEach组件在index对应索引处数据有变化,需要重建该子组件
234  notifyDataChange(index: number): void {
235    this.listeners.forEach(listener => {
236      listener.onDataChange(index);
237    })
238  }
239
240  // 通知LazyForEach组件需要在index对应索引处删除该子组件
241  notifyDataDelete(index: number): void {
242    this.listeners.forEach(listener => {
243      listener.onDataDelete(index);
244    })
245  }
246}
247
248class MyDataSource extends BasicDataSource {
249  private dataArray: string[] = [];
250
251  public totalCount(): number {
252    return this.dataArray.length;
253  }
254
255  public getData(index: number): string {
256    return this.dataArray[index];
257  }
258
259  public addData(index: number, data: string): void {
260    this.dataArray.splice(index, 0, data);
261    this.notifyDataAdd(index);
262  }
263
264  public pushData(data: string): void {
265    this.dataArray.push(data);
266    this.notifyDataAdd(this.dataArray.length - 1);
267  }
268}
269
270@Entry
271@Component
272struct LforEachTest {
273  private data: MyDataSource = new MyDataSource();
274  @State @Watch("onMessageUpdated") message: number = 0;
275
276  onMessageUpdated() {
277    console.info(`LazyforEach message callback func ${this.message}`)
278  }
279
280  aboutToAppear() {
281    for (let i = 0; i <= 20; i++) {
282      this.data.pushData(`Hello ${i}`)
283    }
284  }
285
286  build() {
287    Column() {
288      Button('change message').onClick(() => {
289        this.message++
290      })
291      List({ space: 3 }) {
292        LazyForEach(this.data, (item: string) => {
293          ListItem() {
294            FreezeChild({ message: this.message, index: item })
295          }
296        }, (item: string) => item)
297      }.cachedCount(5).height(500)
298    }
299
300  }
301}
302
303@Component({ freezeWhenInactive: true })
304struct FreezeChild {
305  @Link @Watch("onMessageUpdated") message: number;
306  private index: string = "";
307
308  aboutToAppear() {
309    console.info(`FreezeChild aboutToAppear index: ${this.index}`)
310  }
311
312  onMessageUpdated() {
313    console.info(`FreezeChild message callback func ${this.message}, index: ${this.index}`)
314  }
315
316  build() {
317    Text("message" + `${this.message}, index: ${this.index}`)
318      .width('90%')
319      .height(160)
320      .backgroundColor(0xAFEEEE)
321      .textAlign(TextAlign.Center)
322      .fontSize(30)
323      .fontWeight(FontWeight.Bold)
324  }
325}
326```
327
328在上面的示例中:
329
3301.点击“change message”更改message的值,当前正在显示的ListItem中的子组件@Watch中注册的方法onMessageUpdated被触发。缓存节点@Watch中注册的方法不会被触发。(如果不加组件冻结,当前正在显示的ListItem和cachecount缓存节点@Watch中注册的方法onMessageUpdated都会触发watch回调。)
331
3322.List区域外的ListItem滑动到List区域内,状态由inactive变为active,对应的@Watch中注册的方法onMessageUpdated被触发。
333
3343.再次点击“change message”更改message的值,仅有当前显示的ListItem中的子组件@Watch中注册的方法onMessageUpdated被触发。
335
336![FrezzeLazyforEach.gif](figures/FrezzeLazyforEach.gif)
337
338### Navigation
339
340- 当NavDestination不可见时,会对其子自定义组件设置成非激活态,不会触发组件的刷新。当返回该页面时,其子自定义组件重新恢复成激活态,触发@Watch回调进行刷新。
341
342- 在下面例子中,NavigationContentMsgStack会被设置成非激活态,将不再响应状态变量的变化,也不会触发组件刷新。
343
344```ts
345@Entry
346@Component
347struct MyNavigationTestStack {
348  @Provide('pageInfo') pageInfo: NavPathStack = new NavPathStack();
349  @State @Watch("info") message: number = 0;
350  @State logNumber: number = 0;
351
352  info() {
353    console.info(`freeze-test MyNavigation message callback ${this.message}`);
354  }
355
356  @Builder
357  PageMap(name: string) {
358    if (name === 'pageOne') {
359      pageOneStack({ message: this.message, logNumber: this.logNumber })
360    } else if (name === 'pageTwo') {
361      pageTwoStack({ message: this.message, logNumber: this.logNumber })
362    } else if (name === 'pageThree') {
363      pageThreeStack({ message: this.message, logNumber: this.logNumber })
364    }
365  }
366
367  build() {
368    Column() {
369      Button('change message')
370        .onClick(() => {
371          this.message++;
372        })
373      Navigation(this.pageInfo) {
374        Column() {
375          Button('Next Page', { stateEffect: true, type: ButtonType.Capsule })
376            .width('80%')
377            .height(40)
378            .margin(20)
379            .onClick(() => {
380              this.pageInfo.pushPath({ name: 'pageOne' }); //将name指定的NavDestination页面信息入栈
381            })
382        }
383      }.title('NavIndex')
384      .navDestination(this.PageMap)
385      .mode(NavigationMode.Stack)
386    }
387  }
388}
389
390@Component
391struct pageOneStack {
392  @Consume('pageInfo') pageInfo: NavPathStack;
393  @State index: number = 1;
394  @Link message: number;
395  @Link logNumber: number;
396
397  build() {
398    NavDestination() {
399      Column() {
400        NavigationContentMsgStack({ message: this.message, index: this.index, logNumber: this.logNumber })
401        Text("cur stack size:" + `${this.pageInfo.size()}`)
402          .fontSize(30)
403          .fontWeight(FontWeight.Bold)
404        Button('Next Page', { stateEffect: true, type: ButtonType.Capsule })
405          .width('80%')
406          .height(40)
407          .margin(20)
408          .onClick(() => {
409            this.pageInfo.pushPathByName('pageTwo', null);
410          })
411        Button('Back Page', { stateEffect: true, type: ButtonType.Capsule })
412          .width('80%')
413          .height(40)
414          .margin(20)
415          .onClick(() => {
416            this.pageInfo.pop();
417          })
418      }.width('100%').height('100%')
419    }.title('pageOne')
420    .onBackPressed(() => {
421      this.pageInfo.pop();
422      return true;
423    })
424  }
425}
426
427@Component
428struct pageTwoStack {
429  @Consume('pageInfo') pageInfo: NavPathStack;
430  @State index: number = 2;
431  @Link message: number;
432  @Link logNumber: number;
433
434  build() {
435    NavDestination() {
436      Column() {
437        NavigationContentMsgStack({ message: this.message, index: this.index, logNumber: this.logNumber })
438        Text("cur stack size:" + `${this.pageInfo.size()}`)
439          .fontSize(30)
440          .fontWeight(FontWeight.Bold)
441        Button('Next Page', { stateEffect: true, type: ButtonType.Capsule })
442          .width('80%')
443          .height(40)
444          .margin(20)
445          .onClick(() => {
446            this.pageInfo.pushPathByName('pageThree', null);
447          })
448        Button('Back Page', { stateEffect: true, type: ButtonType.Capsule })
449          .width('80%')
450          .height(40)
451          .margin(20)
452          .onClick(() => {
453            this.pageInfo.pop();
454          })
455      }.width('100%').height('100%')
456    }.title('pageTwo')
457    .onBackPressed(() => {
458      this.pageInfo.pop();
459      return true;
460    })
461  }
462}
463
464@Component
465struct pageThreeStack {
466  @Consume('pageInfo') pageInfo: NavPathStack;
467  @State index: number = 3;
468  @Link message: number;
469  @Link logNumber: number;
470
471  build() {
472    NavDestination() {
473      Column() {
474        NavigationContentMsgStack({ message: this.message, index: this.index, logNumber: this.logNumber })
475        Text("cur stack size:" + `${this.pageInfo.size()}`)
476          .fontSize(30)
477          .fontWeight(FontWeight.Bold)
478        Button('Next Page', { stateEffect: true, type: ButtonType.Capsule })
479          .width('80%')
480          .height(40)
481          .margin(20)
482          .onClick(() => {
483            this.pageInfo.pushPathByName('pageOne', null);
484          })
485        Button('Back Page', { stateEffect: true, type: ButtonType.Capsule })
486          .width('80%')
487          .height(40)
488          .margin(20)
489          .onClick(() => {
490            this.pageInfo.pop();
491          })
492      }.width('100%').height('100%')
493    }.title('pageThree')
494    .onBackPressed(() => {
495      this.pageInfo.pop();
496      return true;
497    })
498  }
499}
500
501@Component({ freezeWhenInactive: true })
502struct NavigationContentMsgStack {
503  @Link @Watch("info") message: number;
504  @Link index: number;
505  @Link logNumber: number;
506
507  info() {
508    console.info(`freeze-test NavigationContent message callback ${this.message}`);
509    console.info(`freeze-test ---- called by content ${this.index}`);
510    this.logNumber++;
511  }
512
513  build() {
514    Column() {
515      Text("msg:" + `${this.message}`)
516        .fontSize(30)
517        .fontWeight(FontWeight.Bold)
518      Text("log number:" + `${this.logNumber}`)
519        .fontSize(30)
520        .fontWeight(FontWeight.Bold)
521    }
522  }
523}
524```
525
526在上面的示例中:
527
5281.点击“change message”更改message的值,当前正在显示的MyNavigationTestStack组件中的@Watch中注册的方法info被触发。
529
5302.点击“Next Page”切换到PageOne,创建pageOneStack节点。
531
5323.再次点击“change message”更改message的值,仅pageOneStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。
533
5344.再次点击“Next Page”切换到PageTwo,创建pageTwoStack节点。
535
5365.再次点击“change message”更改message的值,仅pageTwoStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。
537
5386.再次点击“Next Page”切换到PageThree,创建pageThreeStack节点。
539
5407.再次点击“change message”更改message的值,仅pageThreeStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。
541
5428.点击“Back Page”回到PageTwo,此时,仅pageTwoStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。
543
5449.再次点击“Back Page”回到PageOne,此时,仅pageOneStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。
545
54610.再次点击“Back Page”回到初始页,此时,无任何触发。
547
548![navigation-freeze.gif](figures/navigation-freeze.gif)
549
550### 组件复用
551
552<!--RP1-->[组件复用](../performance/component-recycle.md)<!--RP1End-->通过重利用缓存池中已存在的节点,而非创建新节点,来优化UI性能并提升应用流畅度。复用池中的节点尽管未在UI组件树上展示,但是状态变量的更改仍会触发UI刷新。为了解决复用池中组件异常刷新问题,可以使用组件冻结避免复用池中的组件刷新。
553
554#### 组件复用、if和组件冻结混用场景
555下面是组件复用、if组件和组件冻结混合使用场景的例子,if组件绑定的状态变量变化成false时,触发子组件`ChildComponent`的下树,由于`ChildComponent`被标记了组件复用,所以不会被销毁,而是进入复用池,这个时候如果同时开启了组件冻结,则可以使在复用池里不再刷新。
556具体流程如下:
5571. 点击`change flag`,改变`flag`为false:
558    -  被标记\@Reusable的`ChildComponent`组件在下树时,不会被销毁,而是进入复用池,触发aboutToRecycle生命周期,同时设置状态为inactive。
559    - `ChildComponent`同时也开启了组件冻结,当其状态为inactive时,不会响应任何状态变量变化带来的UI刷新。
5602. 点击`change desc`,触发`Page`的成员变量`desc`的变化。
561    - `desc`是\@State装饰的,其变化会通知给其子组件`ChildComponent`\@Link装饰的`desc`。
562    - 但因为`ChildComponent`是inactive状态,且开启了组件冻结,所以这次变化并不会触发`@Watch('descChange')`的回调,以及`ChildComponent`UI刷新。如果没有开启组件冻结,当前`@Watch('descChange')`会立即回调,且复用池内的`ChildComponent`组件也会对应刷新。
5633. 再次点击`change flag`,改变`flag`为true:
564    - `ChildComponent`从复用池中重新加入到组件树上。
565    - 回调aboutToReuse生命周期,将当前最新的`count`值同步给子组件。`desc`是通过@State->@Link同步的,所以无需开发者手动在aboutToReuse中赋值。
566    - 设置ChildComponent为active状态,并且刷新在inactive时没有刷新的组件,在当前例子中,就是Text(ChildComponent desc: ${this.desc})。
567
568
569```ts
570@Reusable
571@Component({freezeWhenInactive: true})
572struct ChildComponent {
573  @Link @Watch('descChange') desc: string;
574  @State count: number = 0;
575  descChange() {
576    console.info(`ChildComponent messageChange ${this.desc}`);
577  }
578
579  aboutToReuse(params: Record<string, ESObject>): void {
580    this.count = params.count as number;
581  }
582
583  aboutToRecycle(): void {
584    console.info(`ChildComponent has been recycled`);
585  }
586  build() {
587    Column() {
588      Text(`ChildComponent desc: ${this.desc}`)
589        .fontSize(20)
590      Text(`ChildComponent count ${this.count}`)
591        .fontSize(20)
592    }.border({width: 2, color: Color.Pink})
593  }
594}
595
596@Entry
597@Component
598struct Page {
599  @State desc: string = 'Hello World';
600  @State flag: boolean = true;
601  @State count: number = 0;
602  build() {
603    Column() {
604      Button(`change desc`).onClick(() => {
605        this.desc += '!';
606      })
607      Button(`change flag`).onClick(() => {
608        this.count++;
609        this.flag =! this.flag;
610      })
611      if (this.flag) {
612        ChildComponent({desc: this.desc, count: this.count})
613      }
614    }
615    .height('100%')
616  }
617}
618```
619#### LazyForEach、组件复用和组件冻结混用场景
620在数据很多的长列表滑动场景下,开发者会使用LazyForEach来按需创建组件,同时配合组件复用降低在滑动过程中因创建和销毁组件带来的开销。
621但是开发者如果根据其复用类型不同,设置了<!--RP2-->[reuseId](../performance/component-recycle.md#接口说明)<!--RP2End-->,或者为了保证滑动性能设置了较大的cacheCount,这就可能使复用池或者LazyForEach缓存较多的节点。
622在这种情况下,如果开发者触发List下所有子节点的刷新,就会带来节点刷新数量过大的问题,这个时候,可以考虑搭配组件冻结使用。
623
624如下面例子:
6251. 滑动到index为14的位置,当前屏幕上可见区域内有15个`ChildComponent`。
6262. 在滑动过程中:
627    - 列表上端的`ChildComponent`滑出可视区域外,此时先进入LazyForEach的缓存区域内,被设置inactive。在滑出LazyForEach区域外后,因为标记了组件复用,所以并不会被析构,会进入复用池,此时再次被设置inactive。
628    - 列表下端LazyForEach的缓存节点会进入List范围内,此时会试图请求创建新的节点进入LazyForEach的缓存,发现有可复用的节点时,从复用池中拿出已有节点,触发aboutToReuse生命周期回调,此时因为节点进入的是LazyForEach的缓存区域,所以其状态依旧是inactive。
6293. 点击`change desc`,触发`Page`的成员变量`desc`的变化。
630    - `desc`是\@State装饰的,其变化会通知给其子组件`ChildComponent`\@Link装饰的`desc`。
631    - 非可视区域内的`ChildComponent`是inactive状态,且开启了组件冻结,所以这次变化只触发可视区域内的15个节点的`@Watch('descChange')`回调,并只刷新对应可视区域内的15个节点。LazyForEach和复用池中的节点并不会刷新,也不会触发\@Watch回调。
632
633图示如下:
634![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如下面的例子所示,FreezeBuildNode中使用了自定义节点[BuilderNode](../reference/apis-arkui/js-apis-arkui-builderNode.md)。BuilderNode可以通过命令式动态挂载组件,而组件冻结又是强依赖父子关系来通知是否开启组件冻结。如果父组件使用组件冻结,且组件树的中间层级上又启用了BuilderNode,则BuilderNode的子组件将无法被冻结。
976
977```
978import { BuilderNode, FrameNode, NodeController, UIContext } from '@kit.ArkUI';
979
980// 定义一个Params类,用于传递参数
981class Params {
982  index: number = 0;
983
984  constructor(index: number) {
985    this.index = index;
986  }
987}
988
989// 定义一个buildNodeChild组件,它包含一个message属性和一个index属性
990@Component
991struct buildNodeChild {
992  @StorageProp("buildNodeTest") @Watch("onMessageUpdated") message: string = "hello world";
993  @State index: number = 0;
994
995  // 当message更新时,调用此方法
996  onMessageUpdated() {
997    console.log(`FreezeBuildNode builderNodeChild message callback func ${this.message},index:${this.index}`);
998  }
999
1000  build() {
1001    Text(`buildNode Child message: ${this.message}`).fontSize(30)
1002  }
1003}
1004
1005// 定义一个buildText函数,它接收一个Params参数并构建一个Column组件
1006@Builder
1007function buildText(params: Params) {
1008  Column() {
1009    buildNodeChild({ index: params.index })
1010  }
1011}
1012
1013// 定义一个TextNodeController类,继承自NodeController
1014class TextNodeController extends NodeController {
1015  private textNode: BuilderNode<[Params]> | null = null;
1016  private index: number = 0;
1017
1018  // 构造函数接收一个index参数
1019  constructor(index: number) {
1020    super();
1021    this.index = index;
1022  }
1023
1024  // 创建并返回一个FrameNode
1025  makeNode(context: UIContext): FrameNode | null {
1026    this.textNode = new BuilderNode(context);
1027    this.textNode.build(wrapBuilder<[Params]>(buildText), new Params(this.index));
1028    return this.textNode.getFrameNode();
1029  }
1030}
1031
1032// 定义一个Index组件,它包含一个message属性和一个data数组
1033@Entry
1034@Component
1035struct Index {
1036  @StorageLink("buildNodeTest") message: string = "hello";
1037  private data: number[] = [0, 1];
1038
1039  build() {
1040    Row() {
1041      Column() {
1042        Button("change").fontSize(30)
1043          .onClick(() => {
1044            this.message += 'a';
1045          })
1046
1047        Tabs() {
1048          ForEach(this.data, (item: number) => {
1049            TabContent() {
1050              FreezeBuildNode({ index: item })
1051            }.tabBar(`tab${item}`)
1052          }, (item: number) => item.toString())
1053        }
1054      }
1055    }
1056    .width('100%')
1057    .height('100%')
1058  }
1059}
1060
1061// 定义一个FreezeBuildNode组件,它包含一个message属性和一个index属性
1062@Component({ freezeWhenInactive: true })
1063struct FreezeBuildNode {
1064  @StorageProp("buildNodeTest") @Watch("onMessageUpdated") message: string = "1111";
1065  @State index: number = 0;
1066
1067  // 当message更新时,调用此方法
1068  onMessageUpdated() {
1069    console.log(`FreezeBuildNode message callback func ${this.message}, index: ${this.index}`);
1070  }
1071
1072  build() {
1073    NodeContainer(new TextNodeController(this.index))
1074      .width('100%')
1075      .height('100%')
1076      .backgroundColor('#FFF0F0F0')
1077  }
1078}
1079```
1080
1081在上面的示例中:
1082
1083点击Button("change")。改变message的值,当前正在显示的TabContent组件中的@Watch中注册的方法onMessageUpdated被触发。未显示的TabContent中的BuilderNode节点下组件的@Watch方法onMessageUpdated也被触发,并没有被冻结。
1084
1085![builderNode.gif](figures/builderNode.gif)
1086
1087