HarmonyOS鸿蒙Next中基于ArkUI页面抛滑白块优化解决方案

发布于 1周前 作者 gougou168 来自 鸿蒙OS

HarmonyOS鸿蒙Next中基于ArkUI页面抛滑白块优化解决方案 简介

使用imageKnife后仍存在滑动白块问题的场景,常规的解决方案是设置更大的cachedCount缓存数量,但这种方案可能会导致首屏白屏和内场占用增多,针对这个问题,本文将主要提供一种动态预加载的方案,首先介绍相关原理,针对两种技术组合即LazyForeEach+ImageKnife、Repeat+ImageKnife,再分别结合prefetch提供对应场景的开发案例,最终对比不同方案的测试数据。

原理介绍

Imageknife原理介绍
ImageKnife是专门为OpenHarmony打造的一款图像加载缓存库,它封装了一套完整的图片加载流程,开发者只需根据ImageKnifeOption配置相关信息即可完成图片的开发,降低了开发难度,提升了开发效率。

详细介绍可参考:https://gitee.com/openharmony-tpc/ImageKnife

LazyForeEach原理介绍
LazyForEach从提供的数据源中按需迭代数据,并在每次迭代过程中创建相应的组件。当在滚动容器中使用了LazyForEach,框架会根据滚动容器可视区域按需创建组件,当组件滑出可视区域外时,框架会进行组件销毁回收以降低内存占用。

详细介绍可参考:https://developer.huawei.com/consumer/cn/doc/harmonyos-references-V5/ts-rendering-control-lazyforeach-V5

Repeat原理介绍
Repeat组件不开启virtualScroll开关时,Repeat基于数组类型数据来进行循环渲染,需要与容器组件配合使用,且接口返回的组件应当是允许包含在Repeat父容器组件中的子组件。Repeat循环渲染和ForEach相比有两个区别,一是优化了部分更新场景下的渲染性能,二是组件生成函数的索引index由框架侧来维护。

Repeat组件开启virtualScroll开关时,Repeat将从提供的数据源中按需迭代数据,并在每次迭代过程中创建相应的组件。当在滚动容器中使用了Repeat,框架会根据滚动容器可视区域按需创建组件,当组件滑出可视区域外时,框架会缓存组件,并在下一次迭代中使用。

详细介绍可参考:https://developer.huawei.com/consumer/cn/doc/harmonyos-references-V5/ts-rendering-control-repeat-V5

prefetcher原理介绍
prefetch是一种动态预加载技术,通过考虑滚动速度、屏幕上的项目数量等因素,动态的下载或取消下载资源,确保相关资源在需要时能立即显示。

详细介绍可参考:[https://ohpm.openharmony.cn/#/cn/detail/@netteam%2Fprefetcher](https://ohpm.openharmony.cn/#/cn/detail/@netteam%2Fprefetcher)

页面抛滑白块优化解决方案原理介绍

使用LazyForEach/Repeat遍历数据项,通过实现Prefetcher接口监听数据项,选择合适的时机预取数据,使用ImageKnife三方库实现具体的预取功能,并管理缓存。

场景案例

本解决方案针对两种技术组合即LazyForeEach+ImageKnife+prefetch(首页)、Repeat+ImageKnife+prefetch(分类),其中提供对应场景的开发案例,界面效果。

关键代码如下:

1. Prefetcher结合LazyForeEach实现瀑布流页面关键代码

private readonly dataSource = new DataSource(); // 创建数据源 
private readonly prefetcher = createPrefetcher() // 创建prefetcher 
  .withDataSource(this.dataSource) // 绑定数据源 
  .withAddItemsCallback(() => { // 增加数据源的回调函数 
    this.dataCount = this.dataSource.totalCount(); 
    if (this.addItemsCount < 20) { 
      this.addItemsCount ++; 
      setTimeout(() => { 
        this.dataSource.batchAdd(this.dataSource.totalCount()); 
      }, 1000); 
    } 
  }) 
  .withAgent(new ImageKnifeWaterFlowInfoFetchingAgent()); // 绑定获取数据源项引用的数据的代理 

build() { 
  Column({ space: CommonConstants.SPACE_EIGHT }) { 
    Column() { 
      WaterFlow({ footer: this.footStyle, scroller: this.waterFlowScroller }) { 
        LazyForEach(this.dataSource, (item: WaterFlowInfoItem) => { // 瀑布流中使用LazyForEach遍历数据源 
          FlowItem() { 
            WaterFlowImageView({ 
              waterFlowInfoItem: item, 
              waterFlowItemWidth: this.waterFlowItemWidth 
            }) 
          } 
          .height(item.waterFlowHeadInfo.height / item.waterFlowHeadInfo.width * this.waterFlowItemWidth + 
          this.getTitleHeight(item.waterFlowDescriptionInfo.title)) // 通过固定宽高比计算卡片的高度 
          .backgroundColor(Color.White) 
          .width($r('app.string.full_screen')) 
          .clip(true) 
          .borderRadius($r('app.float.rounded_size_16')) 
        }); 
      } 
      .cachedCount(3) 
      .onVisibleAreaChange([0.0, 1.0], (isVisible: boolean) => { 
        // 根据瀑布流卡片可见区域变化,调用prefetch的start()和stop()接口 
        if (isVisible) { 
          this.prefetcher.start(); 
        } else { 
          this.prefetcher.stop(); 
        } 
      }) 
      .onScrollIndex((start: number, end: number) => { 
        // 列表滚动触发visibleAreaChanged,实时更新预取范围,触发调用prefetch、cancel接口 
        this.prefetcher.visibleAreaChanged(start, end); 
      }) 
      .nestedScroll({ scrollForward: NestedScrollMode.PARENT_FIRST, scrollBackward: NestedScrollMode.SELF_FIRST }) 
      .onReachEnd(() => { 
        this.listenNetworkEvent(); 
      }) 
      .columnsTemplate(new BreakpointType({ 
        sm: BreakpointConstants.GRID_NUM_TWO, 
        md: BreakpointConstants.GRID_NUM_THREE, 
        lg: BreakpointConstants.GRID_NUM_FOUR 
      }).getValue(this.currentBreakpoint)) 
      .columnsGap($r('app.float.water_flow_column_gap')) 
      .rowsGap($r('app.float.water_flow_row_gap')) 
      .layoutDirection(FlexDirection.Column) 
      .itemConstraintSize({ 
        minWidth: $r('app.string.zero_screen'), 
        maxWidth: $r('app.string.full_screen'), 
        minHeight: $r('app.string.zero_screen'), 
      }); 
    } 
    .width($r('app.string.full_screen')) 
    .height($r('app.string.full_screen')); 
  } 
  .height($r('app.string.full_screen')) 
  .margin({ 
    top: $r('app.float.margin_8'), 
    bottom: $r('app.float.navigation_height'), 
    left: new BreakpointType({ 
      sm: BreakpointConstants.SEARCHBAR_AND_WATER_FLOW_MARGIN_LEFT_SM, 
      md: BreakpointConstants.SEARCHBAR_AND_WATER_FLOW_MARGIN_LEFT_MD, 
      lg: BreakpointConstants.SEARCHBAR_AND_WATER_FLOW_MARGIN_LEFT_LG 
    }).getValue(this.currentBreakpoint), 
    right: new BreakpointType({ 
      sm: BreakpointConstants.SEARCHBAR_AND_WATER_FLOW_MARGIN_RIGHT_SM, 
      md: BreakpointConstants.SEARCHBAR_AND_WATER_FLOW_MARGIN_RIGHT_MD, 
      lg: BreakpointConstants.SEARCHBAR_AND_WATER_FLOW_MARGIN_RIGHT_LG 
    }).getValue(this.currentBreakpoint) 
  }) 
  .animation({ 
    duration: CommonConstants.ANIMATION_DURATION_TIME, 
    curve: Curve.EaseOut, 
    playMode: PlayMode.Normal 
  }); 
}

2. Prefetcher结合Repeat实现瀑布流页面关键代码

@Local private readonly items: WaterFlowInfoItem[] = wrapArray([]); 
private prefetcher = createPrefetcher() 
  .withDataSource(this.items) // 绑定数据源 
  .withAddItemsCallback(async () => { // 新增数据回调 
    this.pageIndex = (this.pageIndex++) % CommonConstants.REPEAT_WATER_FLOW_PAGES; 
    const waterFlowInfoItemArray = 
      await this.addData(CommonConstants.MOCK_INTERFACE_WATER_FLOW_FILE_NAME, 
        this.pageIndex, CommonConstants.WATER_FLOW_PAGE_SIZE); 
    this.items.push(...waterFlowInfoItemArray); 
  }) 
  .withAgent(new ImageKnifeWaterFlowInfoFetchingAgent()); // 预加载资源代理 

build() { 
  Column() { 
    WaterFlow({ footer: this.footStyle, scroller: this.waterFlowScroller }) { 
      Repeat<WaterFlowInfoItem>(this.items) 
        .each((obj: RepeatItem<WaterFlowInfoItem>) => { 
          FlowItem() { 
            WaterFlowItemComponent({ waterFlowInfoItem: obj.item, waterFlowItemWidth: this.waterFlowItemWidth }) 
          } 
          .height(obj.item.waterFlowHeadInfo.height / obj.item.waterFlowHeadInfo.width * this.waterFlowItemWidth + 
          this.getTitleHeight(obj.item.waterFlowDescriptionInfo.title)) 
          .backgroundColor(Color.White) 
          .width($r('app.string.full_screen')) 
          .clip(true) 
          .borderRadius($r('app.float.rounded_size_16')) 
        }) 
        .key((item: WaterFlowInfoItem) => { 
          return item.key; 
        }) 
    } 
    .cachedCount(3) 
    .onVisibleAreaChange([0.0, 1.0], (isVisible: boolean) => { 
      // 根据瀑布流卡片可见区域变化,调用prefetch的start()和stop()接口 
      if (isVisible) { 
        this.prefetcher.start(); 
      } else { 
        this.prefetcher.stop(); 
      } 
    }) 
    .onScrollIndex((start: number, end: number) => { 
      // 列表滚动触发visibleAreaChanged,实时更新预取范围,触发调用prefetch、cancel接口 
      this.prefetcher.visibleAreaChanged(start, end); 
    }) 
    .nestedScroll({ scrollForward: NestedScrollMode.PARENT_FIRST, scrollBackward: NestedScrollMode.SELF_FIRST }) 
    .onReachEnd(() => { 
      this.listenNetworkEvent(); 
    }) 
    .columnsTemplate(new BreakpointType({ 
      sm: BreakpointConstants.GRID_NUM_TWO, 
      md: BreakpointConstants.GRID_NUM_THREE, 
      lg: BreakpointConstants.GRID_NUM_FOUR 
    }).getValue(this.currentBreakpoint)) 
    .columnsGap($r('app.float.water_flow_column_gap')) 
    .rowsGap($r('app.float.water_flow_row_gap')) 
    .layoutDirection(FlexDirection.Column) 
    .itemConstraintSize({ 
      minWidth: $r('app.string.zero_screen'), 
      maxWidth: $r('app.string.full_screen'), 
      minHeight: $r('app.string.zero_screen'), 
    }); 
  } 
  .height($r('app.string.full_screen')) 
  .margin({ 
    top: $r('app.float.margin_8'), 
    bottom: $r('app.float.navigation_height'), 
    left: new BreakpointType({ 
      sm: BreakpointConstants.SEARCHBAR_AND_WATER_FLOW_MARGIN_LEFT_SM, 
      md: BreakpointConstants.SEARCHBAR_AND_WATER_FLOW_MARGIN_LEFT_MD, 
      lg: BreakpointConstants.SEARCHBAR_AND_WATER_FLOW_MARGIN_LEFT_LG 
    }).getValue(this.currentBreakpoint), 
    right: new BreakpointType({ 
      sm: BreakpointConstants.SEARCHBAR_AND_WATER_FLOW_MARGIN_RIGHT_SM, 
      md: BreakpointConstants.SEARCHBAR_AND_WATER_FLOW_MARGIN_RIGHT_MD, 
      lg: BreakpointConstants.SEARCHBAR_AND_WATER_FLOW_MARGIN_RIGHT_LG 
    }).getValue(this.currentBreakpoint) 
  }) 
  .animation({ 
    duration: CommonConstants.ANIMATION_DURATION_TIME, 
    curve: Curve.EaseOut, 
    playMode: PlayMode.Normal 
  }); 
}

3. Prefetcher必须要实现的接口

IDataReferenceItem接口:要与Prefetcher一起使用的数据源项的接口。用作预取器数据源的数据源项或数组元素实现此接口。

核心代码:

export type FetchParameters = string; 
export type PathToResultFile = string; 
const IMAGE_UNAVAILABLE = $r('app.media.default_image'); 

@Observed 
export class WaterFlowInfoItem implements IDataReferenceItem<FetchParameters, PathToResultFile> { 
  private static nextKey = -1; 
  private _key: number = WaterFlowInfoItem.getKey(); 
 
  private static getKey() { 
    return ++WaterFlowInfoItem.nextKey; 
  } 
 
  public waterFlowHeadInfo: WaterFlowHeadInfo; 
  public waterFlowDescriptionInfo: WaterFlowDescriptionInfo; 
  cachedImage: ResourceStr = ''; 
 
  get key(): string { 
    return this._key.toString(); 
  } 
 
  regenerateKey() { 
    this._key = WaterFlowInfoItem.getKey(); 
  } 
 
  constructor(info: WaterFlowInfo) { 
    this.waterFlowHeadInfo = info.waterFlowHead; 
    this.waterFlowDescriptionInfo = info.waterFlowDescription; 
  } 
 // 预取完成时的回调函数 
  onFetchDone(result: PathToResultFile): void { 
    this.cachedImage = result; 
  } 
 // 预取失败时的回调函数 
  onFetchFail(_details: Error): void { 
    this.cachedImage = IMAGE_UNAVAILABLE; 
  } 
 // 获取需要预取的资源链接 
  getFetchParameters(): FetchParameters { 
    return this.waterFlowHeadInfo.source; 
  } 
 // 判断是否需要预取 
  hasToFetch(): boolean { 
    return !this.cachedImage; 
  } 
}

ITypedDataSource接口:可以链接到Prefetcher的数据源的接口。不需要修改数据源的实际实现。该接口确保数据源项类型与获取代理类型 IFetchAgent 匹配。

核心代码:

export class DataSource implements ITypedDataSource<WaterFlowInfoItem> { 
  private data: WaterFlowInfoItem[] = []; 
  private readonly notifier: Notifier; 
  private netWorkUtil: NetworkUtil = new NetworkUtil(); 
  private pageIndex: number = CommonConstants.NUMBER_DEFAULT_VALUE 
 
  constructor(notificationMode: NotificationMode = 'data-set-changed-method') { 
    this.notifier = new Notifier(notificationMode); 
  } 
 
// 具体的添加数据方法 
  async addData(fileName: string, pageNo: number, pageSize: number): Promise<WaterFlowInfoItem[]> { 
    let waterFlowInfoArray = 
      await this.netWorkUtil.getWaterFlowData(CommonConstants.MOCK_INTERFACE_PATH_NAME, fileName, pageNo, pageSize); 
 
    for (let index = 0; index < waterFlowInfoArray.length; index++) { 
      if (waterFlowInfoArray[index].waterFlowDescription.userName.length > CommonConstants.MAX_NAME_LENGTH) { 
        waterFlowInfoArray[index].waterFlowDescription.userName = 
          waterFlowInfoArray[index].waterFlowDescription.userName.substring(0, CommonConstants.MAX_NAME_LENGTH); 
      } 
      this.data.push(new WaterFlowInfoItem(waterFlowInfoArray[index])); 
    } 
    return this.data; 
  } 
 
// 外部调用的添加数据的方法 
  async batchAdd(startIndex: number) { 
    this.pageIndex = (this.pageIndex++) % CommonConstants.LAZY_FOREACH_WATER_FLOW_PAGES; 
    const items = 
      await this.addData(CommonConstants.MOCK_INTERFACE_WATER_FLOW_FILE_NAME, 
        this.pageIndex, CommonConstants.WATER_FLOW_PAGE_SIZE); 
    this.data.splice(startIndex, 0, ...items); 
    this.notifier.notifyBatchUpdate([ 
      { 
        type: DataOperationType.ADD, 
        index: startIndex, 
        count: items.length, 
        key: items.map(item => item.key) 
      } 
    ]); 
  } 
 
  getData(index: number): WaterFlowInfoItem { 
    return this.data[index]; 
  } 
 
  totalCount(): number { 
    return this.data.length; 
  } 
 
  registerDataChangeListener(listener: DataChangeListener): void { 
    this.notifier.registerDataChangeListener(listener); 
  } 
 
  unregisterDataChangeListener(listener: DataChangeListener): void { 
    this.notifier.unregisterDataChangeListener(listener); 
  } 
 
  deleteAllAsReload(): void { 
    this.data.length = 0; 
    this.notifier.notifyReloaded(); 
  } 
 
  batchDelete(startIndex: number, count: number) { 
    if (startIndex >= 0 && startIndex < this.data.length) { 
      const deleted = this.data.splice(startIndex, count); 
      this.notifier.notifyBatchUpdate([ 
        { 
          type: DataOperationType.DELETE, 
          index: startIndex, 
          count: deleted.length 
        } 
      ]); 
    } 
  } 
}

IFetchAgent接口:该实现负责获取数据源项引用的数据。预取器构建器 API 确保数据源项 IDataReferenceItem 和绑定到预取器实例的获取代理具有匹配的类型。

核心代码:

/* 
* Implementing IFetchAgent for prefetcher using ImageKnife's caching capability 
*/ 
export class ImageKnifeWaterFlowInfoFetchingAgent implements IFetchAgent<FetchParameters, PathToResultFile> { 
  private readonly logger = new Logger("FetchAgent"); 
  private readonly fetchToRequestMap: HashMap<FetchId, ImageKnifeRequest> = new HashMap() 
 
 /* 
 * Asynchronous prefetching function encapsulated with ImageKnife 
 */ 
 async prefetch(fetchId: FetchId, loadSrc: string): Promise<string> { 
    return new Promise((resolve, reject) => { 
      let imageKnifeOption = new ImageKnifeOption() 
      if (typeof loadSrc == 'string') { 
        imageKnifeOption.loadSrc = loadSrc; 
      } else { 
        imageKnifeOption = loadSrc; 
      } 
 
      imageKnifeOption.onLoadListener = { 
        onLoadSuccess: () => { 
          this.fetchToRequestMap.remove(fetchId); 
          resolve(loadSrc); 
        }, 
        onLoadFailed: (err) => { 
          this.fetchToRequestMap.remove(fetchId); 
          reject(err); 
        } 
      } 
      let request = ImageKnife.getInstance().preload(imageKnifeOption); 
      this.fetchToRequestMap.set(fetchId, request); 
    }) 
  } 
 
// 实现prefetcher的fetch接口 
 async fetch(fetchId: FetchId, fetchParameters: FetchParameters): Promise<PathToResultFile> { 
    this.logger.debug(`Fetch ${fetchId}`); 
    let path = await this.prefetch(fetchId, fetchParameters); 
    return path; 
  } 
 
// 实现prefetcher的cancel接口 
  cancel(fetchId: FetchId) { 
    this.logger.debug(`Fetch ${fetchId} cancel`); 
    if (this.fetchToRequestMap.hasKey(fetchId)) { 
      const request = this.fetchToRequestMap.get(fetchId); 
      ImageKnife.getInstance().cancel(request); 
      this.fetchToRequestMap.remove(fetchId); 
    } 
  } 
}

性能分析

本案例中的页面一屏大概可以加载4至6条数据,每张图片的大小在200-300KB之间。针对使用LazyForeEach的场景,使用prefetch方案前后,快速滑动场景下的白块的数量效果对比如下:

使用prefetch方案前 使用prefetch方案后

在快速滑动的场景下,因为动态预加载能取消对快速划过的图片的数据请求,节省了大量的网络资源,从而减少了白块的数量。

示例代码

white_block_solution


更多关于HarmonyOS鸿蒙Next中基于ArkUI页面抛滑白块优化解决方案的实战系列教程也可以访问 https://www.itying.com/category-93-b0.html

3 回复

white_block_solution 这个 demo 有问题,api 看起来是之前版本的

更多关于HarmonyOS鸿蒙Next中基于ArkUI页面抛滑白块优化解决方案的实战系列教程也可以访问 https://www.itying.com/category-93-b0.html


在HarmonyOS鸿蒙Next中,基于ArkUI页面抛滑白块优化的解决方案主要涉及以下几个方面:

  1. 页面渲染优化:通过减少不必要的重绘和重排,提升页面渲染效率。ArkUI框架提供了高效的渲染机制,开发者可以通过合理使用Flex布局、Grid布局等布局方式,减少页面元素的频繁更新。

  2. 列表性能优化:在列表滑动场景中,使用LazyForEach组件替代传统的ForEach,实现懒加载,减少一次性加载大量数据带来的性能压力。同时,合理设置ListItem的复用机制,避免频繁创建和销毁组件。

  3. 异步数据处理:在页面滑动过程中,避免在主线程中进行耗时操作。通过TaskDispatcher机制,将数据处理和网络请求等耗时任务放入后台线程执行,确保UI线程的流畅性。

  4. 内存管理:优化内存使用,避免内存泄漏。ArkUI框架提供了自动内存管理机制,开发者需注意及时释放不再使用的资源,避免内存占用过高导致页面卡顿。

  5. 事件处理优化:在滑动事件处理中,使用Gesture组件替代传统的onTouch事件,减少事件处理的复杂度。通过GestureonPanonFling等回调,实现更高效的滑动事件处理。

  6. 动画优化:在页面滑动过程中,避免使用复杂的动画效果。ArkUI提供了Animator组件,开发者可以通过设置简单的属性动画或关键帧动画,提升页面滑动的流畅性。

通过这些优化措施,可以有效减少鸿蒙Next中基于ArkUI页面的抛滑白块现象,提升用户体验。

在HarmonyOS鸿蒙Next中,基于ArkUI页面抛滑白块问题,可通过以下优化方案解决:

  1. 减少渲染层级:简化组件结构,避免不必要的嵌套,提升渲染效率。
  2. 使用缓存机制:对频繁更新的组件使用缓存,减少重复渲染。
  3. 异步加载数据:将数据加载与UI渲染分离,避免主线程阻塞。
  4. 优化动画:使用硬件加速动画,减少CPU占用。
  5. 分页加载:对长列表采用分页加载,避免一次性渲染过多内容。
  6. 性能监控:使用DevEco Studio的性能分析工具,定位瓶颈并优化。

通过这些措施,可有效减少页面抛滑时的白块现象,提升用户体验。

回到顶部
AI 助手
你好,我是IT营的 AI 助手
您可以尝试点击下方的快捷入口开启体验!