0


HarmonyOS:自定义组件冻结功能

一、简介

自定义组件冻结功能专为优化复杂UI页面的性能而设计,尤其适用于包含多个页面栈、长列表或宫格布局的场景。在这些情况下,当状态变量绑定了多个UI组件,其变化可能触发大量UI组件的刷新,进而导致界面卡顿和响应延迟。为了提升这类负载UI界面的刷新性能,开发者可以选择尝试使用自定义组件冻结功能。

组件冻结的工作原理是:

  1. 开发者通过设置freezeWhenInactive属性,即可激活组件冻结机制。
  2. 启用后,系统将仅对处于激活状态的自定义组件进行更新,这使得UI框架可以尽量缩小更新范围,仅限于用户可见范围内(激活状态)的自定义组件,从而提高复杂UI场景下的刷新效率。
  3. 当之前处于inactive状态的自定义组件重新变为active状态时,状态管理框架会对其执行必要的刷新操作,确保UI的正确展示。

简而言之,组件冻结旨在优化复杂界面下的UI刷新性能。在存在多个不可见自定义组件的情况下,如多页面栈、长列表或宫格,通过组件冻结可以实现按需刷新,即仅刷新当前可见的自定义组件,而将不可见自定义组件的刷新延迟至它们变为可见时。

需要注意,组件active/inactive并不等同于其可见性。组件冻结目前仅适用于以下场景:

  1. 页面路由:当前栈顶页面为active,非栈顶不可见页面为inactive
  2. TabContent:只有当前显示的TabContent中的自定义组件处于active状态,其余则为inactive。
  3. LazyForEach:仅当前显示的LazyForEach中的自定义组件为active,而缓存节点的组件则为inactive。
  4. Navigation:当前显示的NavDestination中的自定义组件为active,而其他未显示的NavDestination组件则为inactive。 其他场景,如堆叠布局(Stack)下的被遮罩的组件,这些组件尽管不可见,但并不被视为inactive状态,因此不在组件冻结的适用范围内。

说明
从API version 11开始,支持自定义组件冻结功能。

二、当前支持的场景

2.1 页面路由
  • 当页面A调用router.pushUrl接口跳转到页面B时,页面A为隐藏不可见状态,此时如果更新页面A中的状态变量,不会触发页面A刷新。页面A
  1. import{ router } from '@kit.ArkUI';
  2. @Entry
  3. @Component({ freezeWhenInactive: true})
  4. struct FirstTest {
  5. @StorageLink('PropA') @Watch("first") storageLink: number =47;first(){
  6. console.info("first page " + `${this.storageLink}`)}build(){Column(){
  7. Text(`From fist Page ${this.storageLink}`).fontSize(30)
  8. Button('first page storageLink + 1').fontSize(20).margin({ top: 16})
  9. .onClick(()=>{
  10. this.storageLink +=1})
  11. Button('go to next page').fontSize(30)
  12. .margin({ top: 16})
  13. .onClick(()=>{
  14. router.pushUrl({ url: 'pages/SecondTest'})})}}}

页面B

  1. import{ router } from '@kit.ArkUI';
  2. @Entry
  3. @Component({ freezeWhenInactive: true})
  4. struct SecondTest {
  5. @StorageLink('PropA') @Watch("second") storageLink2: number =1;second(){
  6. console.info("second page: " + `${this.storageLink2}`)}build(){Column(){
  7. Text(`second Page ${this.storageLink2}`).fontSize(50)
  8. Button('Change Divider.strokeWidth')
  9. .onClick(()=>{
  10. router.back()})
  11. Button('second page storageLink2 + 2').fontSize(30)
  12. .onClick(()=>{
  13. this.storageLink2 +=2})}}}

在上面的示例中:

  1. 点击页面A中的Button “first page storageLink + 1”,storageLink状态变量改变,@Watch中注册的方法first会被调用。
  2. 通过router.pushUrl({url: ‘pages/second’}),跳转到页面B,页面A隐藏,状态由active变为inactive。 3 .点击页面B中的Button “this.storageLink2 += 2”,只回调页面B@Watch中注册的方法second,因为页面A的状态变量此时已被冻结。
  3. 点击“back”,页面B被销毁,页面A的状态由inactive变为active,重新刷新在inactive时被冻结的状态变量,页面A@Watch中注册的方法first被再次调用。
2.2 TabContent

对Tabs中当前不可见的TabContent进行冻结,不会触发组件的更新。

需要注意的是:在首次渲染的时候,Tab只会创建当前正在显示的TabContent,当切换全部的TabContent后,TabContent才会被全部创建。

  1. @Entry
  2. @Component
  3. struct TabContentTest {
  4. @State @Watch("onMessageUpdated") message: number =0;
  5. private data: number[]=[0, 1]onMessageUpdated(){
  6. console.info(`TabContent message callback func ${this.message}`)}build(){Row(){Column(){
  7. Button('change message').margin({ top: 40}).onClick(()=> {
  8. this.message++
  9. })
  10. Tabs() {
  11. ForEach(this.data,(item: number)=> {
  12. TabContent() {
  13. FreezeChild({ message: this.message, index: item })
  14. }.tabBar(`tab${item}`)
  15. },(item: number)=> item.toString())}}
  16. .width('100%')}
  17. .height('100%')}}
  18. @Component({ freezeWhenInactive: true})
  19. struct FreezeChild {
  20. @Link @Watch("onMessageUpdated") message: number
  21. private index: number =0onMessageUpdated(){
  22. console.info(`FreezeChild message callback func ${this.message}, index: ${this.index}`)}build(){
  23. Text("message" + `${this.message}, index: ${this.index}`)
  24. .fontSize(30)
  25. .fontWeight(FontWeight.Bold)}}

在上面的示例中:

  1. 点击“change message”更改message的值,当前正在显示的TabContent组件中的@Watch中注册的方法onMessageUpdated被触发。
  2. 点击“two”切换到另外的TabContent,TabContent状态由inactive变为active,对应的@Watch中注册的方法onMessageUpdated被触发。
  3. 再次点击“change message”更改message的值,仅当前显示的TabContent子组件中的@Watch中注册的方法onMessageUpdated被触发。

运行效果图
在这里插入图片描述

2.3 LazyForEach
  • 对LazyForEach中缓存的自定义组件进行冻结,不会触发组件的更新。
  1. // Basic implementation of IDataSource to handle data listener
  2. class BasicDataSource implements IDataSource {
  3. private listeners: DataChangeListener[]=[];
  4. private originDataArray: string[]=[];
  5. public totalCount(): number {return0;}
  6. public getData(index: number): string {return this.originDataArray[index];}
  7. // 该方法为框架侧调用,为LazyForEach组件向其数据源处添加listener监听
  8. registerDataChangeListener(listener: DataChangeListener): void {if(this.listeners.indexOf(listener)<0){
  9. console.info('add listener');
  10. this.listeners.push(listener);}}
  11. // 该方法为框架侧调用,为对应的LazyForEach组件在数据源处去除listener监听
  12. unregisterDataChangeListener(listener: DataChangeListener): void {
  13. const pos = this.listeners.indexOf(listener);if(pos >=0){
  14. console.info('remove listener');
  15. this.listeners.splice(pos, 1);}}
  16. // 通知LazyForEach组件需要重载所有子组件
  17. notifyDataReload(): void {
  18. this.listeners.forEach(listener =>{
  19. listener.onDataReloaded();})}
  20. // 通知LazyForEach组件需要在index对应索引处添加子组件
  21. notifyDataAdd(index: number): void {
  22. this.listeners.forEach(listener =>{
  23. listener.onDataAdd(index);})}
  24. // 通知LazyForEach组件在index对应索引处数据有变化,需要重建该子组件
  25. notifyDataChange(index: number): void {
  26. this.listeners.forEach(listener =>{
  27. listener.onDataChange(index);})}
  28. // 通知LazyForEach组件需要在index对应索引处删除该子组件
  29. notifyDataDelete(index: number): void {
  30. this.listeners.forEach(listener =>{
  31. listener.onDataDelete(index);})}}
  32. class MyDataSource extends BasicDataSource {
  33. private dataArray: string[]=[];
  34. public totalCount(): number {return this.dataArray.length;}
  35. public getData(index: number): string {return this.dataArray[index];}
  36. public addData(index: number, data: string): void {
  37. this.dataArray.splice(index, 0, data);
  38. this.notifyDataAdd(index);}
  39. public pushData(data: string): void {
  40. this.dataArray.push(data);
  41. this.notifyDataAdd(this.dataArray.length - 1);}}
  42. @Entry
  43. @Component
  44. struct LforEachTest {
  45. private data: MyDataSource = new MyDataSource();
  46. @State @Watch("onMessageUpdated") message: number =0;onMessageUpdated(){
  47. console.info(`LazyforEach message callback func ${this.message}`)}aboutToAppear(){for(let i =0; i <=20; i++){
  48. this.data.pushData(`Hello ${i}`)}}build(){Column(){
  49. Button('change message').margin({ bottom: 16}).onClick(()=>{
  50. this.message++
  51. })
  52. List({ space: 3}){
  53. LazyForEach(this.data, (item: string)=>{ListItem(){
  54. FreezeChild2({ message: this.message, index: item })}}, (item: string)=> item)}.cachedCount(5).height(500)}}}
  55. @Component({ freezeWhenInactive: true})
  56. struct FreezeChild2 {
  57. @Link @Watch("onMessageUpdated") message: number;
  58. private index: string ="";aboutToAppear(){
  59. console.info(`FreezeChild aboutToAppear index: ${this.index}`)}onMessageUpdated(){
  60. console.info(`FreezeChild message callback func ${this.message}, index: ${this.index}`)}build(){
  61. Text("message" + `${this.message}, index: ${this.index}`)
  62. .width('90%')
  63. .height(160)
  64. .backgroundColor(0xAFEEEE)
  65. .textAlign(TextAlign.Center)
  66. .fontSize(20)
  67. .fontWeight(FontWeight.Bold)}}

在上面的示例中:
1.点击“change message”更改message的值,当前正在显示的ListItem中的子组件@Watch中注册的方法onMessageUpdated被触发。缓存节点@Watch中注册的方法不会被触发。(如果不加组件冻结,当前正在显示的ListItem和cachecount缓存节点@Watch中注册的方法onMessageUpdated都会触发watch回调。)
在这里插入图片描述

2.List区域外的ListItem滑动到List区域内,状态由inactive变为active,对应的@Watch中注册的方法onMessageUpdated被触发。
在这里插入图片描述

3.再次点击“change message”更改message的值,仅有当前显示的ListItem中的子组件@Watch中注册的方法onMessageUpdated被触发。
在这里插入图片描述

2.4 Navigation
  • 当NavDestination不可见时,会对其子自定义组件设置成非激活态,不会触发组件的刷新。当返回该页面时,其子自定义组件重新恢复成激活态,触发@Watch回调进行刷新。
  • 在下面例子中,NavigationContentMsgStack会被设置成非激活态,将不再响应状态变量的变化,也不会触发组件刷新。
  1. @Entry
  2. @Component
  3. struct MyNavigationTestStack {
  4. @Provide('pageInfo') pageInfo: NavPathStack = new NavPathStack();
  5. @State @Watch("info") message: number =0;
  6. @State logNumber: number =0;info(){
  7. console.info(`freeze-test MyNavigation message callback ${this.message}`);}
  8. @Builder
  9. PageMap(name: string){if(name ==='pageOne'){
  10. pageOneStack({ message: this.message, logNumber: this.logNumber })}elseif(name ==='pageTwo'){
  11. pageTwoStack({ message: this.message, logNumber: this.logNumber })}elseif(name ==='pageThree'){
  12. pageThreeStack({ message: this.message, logNumber: this.logNumber })}}build(){Column(){
  13. Button('change message')
  14. .onClick(()=>{
  15. this.message++;})
  16. Navigation(this.pageInfo){Column(){
  17. Button('Next Page', { stateEffect: true, type: ButtonType.Capsule })
  18. .width('80%')
  19. .height(40)
  20. .margin(20)
  21. .onClick(()=>{
  22. this.pageInfo.pushPath({ name: 'pageOne'}); //将name指定的NavDestination页面信息入栈
  23. })}}.title('NavIndex')
  24. .navDestination(this.PageMap)
  25. .mode(NavigationMode.Stack)}}}
  26. @Component
  27. struct pageOneStack {
  28. @Consume('pageInfo') pageInfo: NavPathStack;
  29. @State index: number =1;
  30. @Link message: number;
  31. @Link logNumber: number;build(){NavDestination(){Column(){
  32. NavigationContentMsgStack({ message: this.message, index: this.index, logNumber: this.logNumber })
  33. Text("cur stack size:" + `${this.pageInfo.size()}`)
  34. .fontSize(30)
  35. .fontWeight(FontWeight.Bold)
  36. Button('Next Page', { stateEffect: true, type: ButtonType.Capsule })
  37. .width('80%')
  38. .height(40)
  39. .margin(20)
  40. .onClick(()=>{
  41. this.pageInfo.pushPathByName('pageTwo', null);})
  42. Button('Back Page', { stateEffect: true, type: ButtonType.Capsule })
  43. .width('80%')
  44. .height(40)
  45. .margin(20)
  46. .onClick(()=>{
  47. this.pageInfo.pop();})}.width('100%').height('100%')}.title('pageOne')
  48. .onBackPressed(()=>{
  49. this.pageInfo.pop();returntrue;})}}
  50. @Component
  51. struct pageTwoStack {
  52. @Consume('pageInfo') pageInfo: NavPathStack;
  53. @State index: number =2;
  54. @Link message: number;
  55. @Link logNumber: number;build(){NavDestination(){Column(){
  56. NavigationContentMsgStack({ message: this.message, index: this.index, logNumber: this.logNumber })
  57. Text("cur stack size:" + `${this.pageInfo.size()}`)
  58. .fontSize(20)
  59. .fontWeight(FontWeight.Bold)
  60. Button('Next Page', { stateEffect: true, type: ButtonType.Capsule })
  61. .width('80%')
  62. .height(40)
  63. .margin(20)
  64. .onClick(()=>{
  65. this.pageInfo.pushPathByName('pageThree', null);})
  66. Button('Back Page', { stateEffect: true, type: ButtonType.Capsule })
  67. .width('80%')
  68. .height(40)
  69. .margin(20)
  70. .onClick(()=>{
  71. this.pageInfo.pop();})}.width('100%').height('100%')}.title('pageTwo')
  72. .onBackPressed(()=>{
  73. this.pageInfo.pop();returntrue;})}}
  74. @Component
  75. struct pageThreeStack {
  76. @Consume('pageInfo') pageInfo: NavPathStack;
  77. @State index: number =3;
  78. @Link message: number;
  79. @Link logNumber: number;build(){NavDestination(){Column(){
  80. NavigationContentMsgStack({ message: this.message, index: this.index, logNumber: this.logNumber })
  81. Text("cur stack size:" + `${this.pageInfo.size()}`)
  82. .fontSize(20)
  83. .fontWeight(FontWeight.Bold)
  84. Button('Next Page', { stateEffect: true, type: ButtonType.Capsule })
  85. .width('80%')
  86. .height(40)
  87. .margin(20)
  88. .onClick(()=>{
  89. this.pageInfo.pushPathByName('pageOne', null);})
  90. Button('Back Page', { stateEffect: true, type: ButtonType.Capsule })
  91. .width('80%')
  92. .height(40)
  93. .margin(20)
  94. .onClick(()=>{
  95. this.pageInfo.pop();})}.width('100%').height('100%')}.title('pageThree')
  96. .onBackPressed(()=>{
  97. this.pageInfo.pop();returntrue;})}}
  98. @Component({ freezeWhenInactive: true})
  99. struct NavigationContentMsgStack {
  100. @Link @Watch("info") message: number;
  101. @Link index: number;
  102. @Link logNumber: number;info(){
  103. console.info(`freeze-test NavigationContent message callback ${this.message}`);
  104. console.info(`freeze-test ---- called by content ${this.index}`);
  105. this.logNumber++;}build(){Column(){
  106. Text("msg:" + `${this.message}`)
  107. .fontSize(20)
  108. .fontWeight(FontWeight.Bold)
  109. Text("log number:" + `${this.logNumber}`)
  110. .fontSize(20)
  111. .fontWeight(FontWeight.Bold)}}}

在上面的示例中:

1.点击“change message”更改message的值,当前正在显示的MyNavigationTestStack组件中的@Watch中注册的方法info被触发。
在这里插入图片描述

2.点击“Next Page”切换到PageOne,创建pageOneStack节点。
在这里插入图片描述

3.再次点击“change message”更改message的值,仅pageOneStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。
在这里插入图片描述

4.再次点击“Next Page”切换到PageTwo,创建pageTwoStack节点。
在这里插入图片描述

5.再次点击“change message”更改message的值,仅pageTwoStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。
在这里插入图片描述

6.再次点击“Next Page”切换到PageThree,创建pageThreeStack节点。
在这里插入图片描述

7.再次点击“change message”更改message的值,仅pageThreeStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。
在这里插入图片描述

8.点击“Back Page”回到PageTwo,此时,仅pageTwoStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。
在这里插入图片描述

9.再次点击“Back Page”回到PageOne,此时,仅pageOneStack中的NavigationContentMsgStack子组件中的@Watch中注册的方法info被触发。
在这里插入图片描述

10.再次点击“Back Page”回到初始页,此时,无任何触发。
在这里插入图片描述


本文转载自: https://blog.csdn.net/ChinaDragon10/article/details/143275732
版权归原作者 ChinaDragonDreamer 所有, 如有侵权,请联系我们删除。

“HarmonyOS:自定义组件冻结功能”的评论:

还没有评论