• HarmonyOS—UI开发性能提升的推荐方法


    注:本文转载自HarmonyOS官网文档

    开发者若使用低性能的代码实现功能场景可能不会影响应用的正常运行,但却会对应用的性能造成负面影响。本章节列举出了一些可提升性能的场景供开发者参考,以避免应用实现上带来的性能劣化。

    使用数据懒加载

    开发者在使用长列表时,如果直接采用循环渲染方式,如下所示,会一次性加载所有的列表元素,一方面会导致页面启动时间过长,影响用户体验,另一方面也会增加服务器的压力和流量,加重系统负

    1. @Entry
    2. @Component
    3. struct MyComponent {
    4.   @State arrnumber[] = Array.from(Array(100), (v,k) =>k); //构造0-99的数组
    5. build() {
    6. List() {
    7. ForEach(this.arr, (item: number) => {
    8. ListItem() {
    9. Text(`item value: ${item}`)
    10. }
    11. }, (item: number) => item.toString())
    12. }
    13. }
    14. }

    上述代码会在页面加载时将100个列表元素全部加载,这并非我们需要的,我们希望从数据源中按需迭代加载数据并创建相应组件,因此需要使用数据懒加载,如下所示:

    1. class BasicDataSource implements IDataSource {
    2. private listeners: DataChangeListener[] = []
    3. public totalCount(): number {
    4. return 0
    5. }
    6. public getData(indexnumber): any {
    7. return undefined
    8. }
    9. registerDataChangeListener(listener: DataChangeListener): void {
    10. if (this.listeners.indexOf(listener) < 0) {
    11.       console.info('add listener')
    12. this.listeners.push(listener)
    13. }
    14. }
    15. unregisterDataChangeListener(listener: DataChangeListener): void {
    16. const pos = this.listeners.indexOf(listener);
    17. if (pos >= 0) {
    18.       console.info('remove listener')
    19. this.listeners.splice(pos, 1)
    20. }
    21. }
    22. notifyDataReload(): void {
    23. this.listeners.forEach(listener => {
    24.       listener.onDataReloaded()
    25. })
    26. }
    27. notifyDataAdd(indexnumber): void {
    28. this.listeners.forEach(listener => {
    29.       listener.onDataAdd(index)
    30. })
    31. }
    32. notifyDataChange(indexnumber): void {
    33. this.listeners.forEach(listener => {
    34.       listener.onDataChange(index)
    35. })
    36. }
    37. notifyDataDelete(indexnumber): void {
    38. this.listeners.forEach(listener => {
    39.       listener.onDataDelete(index)
    40. })
    41. }
    42. notifyDataMove(fromnumbertonumber): void {
    43. this.listeners.forEach(listener => {
    44.       listener.onDataMove(fromto)
    45. })
    46. }
    47. }
    48. class MyDataSource extends BasicDataSource {
    49. private dataArray: string[] = ['item value: 0', 'item value: 1', 'item value: 2']
    50. public totalCount(): number {
    51. return this.dataArray.length
    52. }
    53. public getData(indexnumber): any {
    54. return this.dataArray[index]
    55. }
    56. public addData(indexnumberdatastring): void {
    57. this.dataArray.splice(index, 0data)
    58. this.notifyDataAdd(index)
    59. }
    60. public pushData(datastring): void {
    61. this.dataArray.push(data)
    62. this.notifyDataAdd(this.dataArray.length - 1)
    63. }
    64. }
    65. @Entry
    66. @Component
    67. struct MyComponent {
    68. private data: MyDataSource = new MyDataSource()
    69. build() {
    70. List() {
    71. LazyForEach(this.data, (item: string) => {
    72. ListItem() {
    73. Row() {
    74. Text(item).fontSize(20).margin({ left: 10 })
    75. }
    76. }
    77. .onClick(() => {
    78. this.data.pushData('item value: ' + this.data.totalCount())
    79. })
    80. }, item => item)
    81. }
    82. }
    83. }

    上述代码在页面加载时仅初始化加载三个列表元素,之后每点击一次列表元素,将增加一个列表元素。

    设置List组件的宽高

    在使用Scroll容器组件嵌套List组件加载长列表时,若不指定List的宽高尺寸,则默认全部加载。

    说明

    Scroll嵌套List时:

    • List没有设置宽高,会布局List的所有子组件。
    • List设置宽高,会布局List显示区域内的子组件。
    • List使用ForEach加载子组件时,无论是否设置List的宽高,都会加载所有子组件。
    • List使用LazyForEach加载子组件时,没有设置List的宽高,会加载所有子组件,设置了List的宽高,会加载List显示区域内的子组件。

    1. class BasicDataSource implements IDataSource {
    2. private listeners: DataChangeListener[] = []
    3. public totalCount(): number {
    4. return 0
    5. }
    6. public getData(indexnumber): any {
    7. return undefined
    8. }
    9. registerDataChangeListener(listener: DataChangeListener): void {
    10. if (this.listeners.indexOf(listener) < 0) {
    11.       console.info('add listener')
    12. this.listeners.push(listener)
    13. }
    14. }
    15. unregisterDataChangeListener(listener: DataChangeListener): void {
    16. const pos = this.listeners.indexOf(listener);
    17. if (pos >= 0) {
    18.       console.info('remove listener')
    19. this.listeners.splice(pos, 1)
    20. }
    21. }
    22. notifyDataReload(): void {
    23. this.listeners.forEach(listener => {
    24.       listener.onDataReloaded()
    25. })
    26. }
    27. notifyDataAdd(indexnumber): void {
    28. this.listeners.forEach(listener => {
    29.       listener.onDataAdd(index)
    30. })
    31. }
    32. notifyDataChange(indexnumber): void {
    33. this.listeners.forEach(listener => {
    34.       listener.onDataChange(index)
    35. })
    36. }
    37. notifyDataDelete(indexnumber): void {
    38. this.listeners.forEach(listener => {
    39.       listener.onDataDelete(index)
    40. })
    41. }
    42. notifyDataMove(fromnumbertonumber): void {
    43. this.listeners.forEach(listener => {
    44.       listener.onDataMove(fromto)
    45. })
    46. }
    47. }
    48. class MyDataSource extends BasicDataSource {
    49. private dataArray: Array<string> = new Array(100).fill('test')
    50. public totalCount(): number {
    51. return this.dataArray.length
    52. }
    53. public getData(indexnumber): any {
    54. return this.dataArray[index]
    55. }
    56. public addData(indexnumberdatastring): void {
    57. this.dataArray.splice(index, 0data)
    58. this.notifyDataAdd(index)
    59. }
    60. public pushData(datastring): void {
    61. this.dataArray.push(data)
    62. this.notifyDataAdd(this.dataArray.length - 1)
    63. }
    64. }
    65. @Entry
    66. @Component
    67. struct MyComponent {
    68. private data: MyDataSource = new MyDataSource()
    69. build() {
    70. Scroll() {
    71. List() {
    72. LazyForEach(this.data, (item: stringindexnumber) => {
    73. ListItem() {
    74. Row() {
    75. Text('item value: ' + item + (index + 1)).fontSize(20).margin(10)
    76. }
    77. }
    78. })
    79. }
    80. }
    81. }
    82. }

    因此,此场景下建议设置List子组件的宽高。

    1. class BasicDataSource implements IDataSource {
    2. private listeners: DataChangeListener[] = []
    3. public totalCount(): number {
    4. return 0
    5. }
    6. public getData(indexnumber): any {
    7. return undefined
    8. }
    9. registerDataChangeListener(listener: DataChangeListener): void {
    10. if (this.listeners.indexOf(listener) < 0) {
    11.       console.info('add listener')
    12. this.listeners.push(listener)
    13. }
    14. }
    15. unregisterDataChangeListener(listener: DataChangeListener): void {
    16. const pos = this.listeners.indexOf(listener);
    17. if (pos >= 0) {
    18.       console.info('remove listener')
    19. this.listeners.splice(pos, 1)
    20. }
    21. }
    22. notifyDataReload(): void {
    23. this.listeners.forEach(listener => {
    24.       listener.onDataReloaded()
    25. })
    26. }
    27. notifyDataAdd(indexnumber): void {
    28. this.listeners.forEach(listener => {
    29.       listener.onDataAdd(index)
    30. })
    31. }
    32. notifyDataChange(indexnumber): void {
    33. this.listeners.forEach(listener => {
    34.       listener.onDataChange(index)
    35. })
    36. }
    37. notifyDataDelete(indexnumber): void {
    38. this.listeners.forEach(listener => {
    39.       listener.onDataDelete(index)
    40. })
    41. }
    42. notifyDataMove(fromnumbertonumber): void {
    43. this.listeners.forEach(listener => {
    44.       listener.onDataMove(fromto)
    45. })
    46. }
    47. }
    48. class MyDataSource extends BasicDataSource {
    49. private dataArray: Array<string> = new Array(100).fill('test')
    50. public totalCount(): number {
    51. return this.dataArray.length
    52. }
    53. public getData(indexnumber): any {
    54. return this.dataArray[index]
    55. }
    56. public addData(indexnumberdatastring): void {
    57. this.dataArray.splice(index, 0data)
    58. this.notifyDataAdd(index)
    59. }
    60. public pushData(datastring): void {
    61. this.dataArray.push(data)
    62. this.notifyDataAdd(this.dataArray.length - 1)
    63. }
    64. }
    65. @Entry
    66. @Component
    67. struct MyComponent {
    68. private data: MyDataSource = new MyDataSource()
    69. build() {
    70. Scroll() {
    71. List() {
    72. LazyForEach(this.data, (item: stringindexnumber) => {
    73. ListItem() {
    74. Text('item value: ' + item + (index + 1)).fontSize(20).margin(10)
    75. }.width('100%')
    76. })
    77. }.width('100%').height(500)
    78. }.backgroundColor(Color.Pink)
    79. }
    80. }

    使用条件渲染替代显隐控制

    如下所示,开发者在使用visibility通用属性控制组件的显隐状态时,仍存在组件的重新创建过程,造成性能上的损耗。

    1. @Entry
    2. @Component
    3. struct MyComponent {
    4.   @State isVisible: Visibility = Visibility.Visible;
    5.   build() {
    6.     Column() {
    7.       Button("显隐切换")
    8.         .onClick(() => {
    9.           if (this.isVisible == Visibility.Visible) {
    10.             this.isVisible = Visibility.None
    11.           } else {
    12.             this.isVisible = Visibility.Visible
    13.           }
    14.         })
    15.       Row().visibility(this.isVisible)
    16.         .width(300).height(300).backgroundColor(Color.Pink)
    17.     }.width('100%')
    18.   }
    19. }

    要避免这一问题,可使用if条件渲染代替visibility属性变换,如下所示:

    1. @Entry
    2. @Component
    3. struct MyComponent {
    4.   @State isVisibleboolean = true;
    5. build() {
    6. Column() {
    7. Button("显隐切换")
    8. .onClick(() => {
    9. this.isVisible = !this.isVisible
    10. })
    11. if (this.isVisible) {
    12. Row()
    13. .width(300).height(300).backgroundColor(Color.Pink)
    14. }
    15. }.width('100%')
    16. }
    17. }

    使用Column/Row替代Flex

    由于Flex容器组件默认情况下存在shrink导致二次布局,这会在一定程度上造成页面渲染上的性能劣化。

    1. @Entry
    2. @Component
    3. struct MyComponent {
    4.   build() {
    5.     Flex({ direction: FlexDirection.Column }) {
    6.       Flex().width(300).height(200).backgroundColor(Color.Pink)
    7.       Flex().width(300).height(200).backgroundColor(Color.Yellow)
    8.       Flex().width(300).height(200).backgroundColor(Color.Grey)
    9.     }
    10.   }
    11. }

    上述代码可将Flex替换为Column、Row,在保证实现的页面布局效果相同的前提下避免Flex二次布局带来的负面影响。

    1. @Entry
    2. @Component
    3. struct MyComponent {
    4.   build() {
    5.     Column() {
    6.       Row().width(300).height(200).backgroundColor(Color.Pink)
    7.       Row().width(300).height(200).backgroundColor(Color.Yellow)
    8.       Row().width(300).height(200).backgroundColor(Color.Grey)
    9.     }
    10.   }
    11. }

    减少应用滑动白块

    应用通过增大List/Grid控件的cachedCount参数,调整UI的加载范围。cachedCount表示屏幕外List/Grid预加载item的个数。

    如果需要请求网络图片,可以在item滑动到屏幕显示之前,提前下载好内容,从而减少滑动白块。

    如下是使用cachedCount参数的例子:

    1. @Entry
    2. @Component
    3. struct MyComponent {
    4.   private source: MyDataSource = new MyDataSource();
    5.   build() {
    6.     List() {
    7.       LazyForEach(this.source, item => {
    8.         ListItem() {
    9.           Text("Hello" + item)
    10.             .fontSize(50)
    11.             .onAppear(() => {
    12.               console.log("appear:" + item)
    13.             })
    14.         }
    15.       })
    16.     }.cachedCount(3// 扩大数值appear日志范围会变大
    17.   }
    18. }
    19. class MyDataSource implements IDataSource {
    20.   datanumber[] = [123456789101112131415];
    21.   public totalCount(): number {
    22.     return this.data.length
    23.   }
    24.   public getData(indexnumber): any {
    25.     return this.data[index]
    26.   }
    27.   registerDataChangeListener(listener: DataChangeListener): void {
    28.   }
    29.   unregisterDataChangeListener(listener: DataChangeListener): void {
    30.   }
    31. }

    使用说明:

    cachedCount的增加会增大UI的cpu、内存开销。使用时需要根据实际情况,综合性能和用户体验进行调整。

    点击关注阅读原文,了解更多精彩资讯

  • 相关阅读:
    计算机毕业设计 生活废品回收系统 Vue+SpringBoot+MySQL
    20天拿下华为OD笔试之【模拟】2023B-阿里巴巴找黄金宝箱(1)【欧弟算法】全网注释最详细分类最全的华为OD真题题解
    visual studio code(vs code)历史版本下载
    高效正则匹配工具
    JS 字符串
    ubuntu系统 kubeadm方式搭建k8s集群
    华为机试练习题:HJ51 输出单向链表中倒数第k个结点
    智慧新能源电站远程监控系统总体设计
    MES在注塑制造领域的运用
    前端常用设计模式
  • 原文地址:https://blog.csdn.net/HarmonyOSDev/article/details/132598188