Android官方架构组件Paging-Ex:为分页列表添加Header和Footer

2023-11-20 20:20

本文主要是介绍Android官方架构组件Paging-Ex:为分页列表添加Header和Footer,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本文已授权「玉刚说」微信公众号独家发布

2019/12/24 补充

距本文发布时隔一年,笔者认为,本文不应该作为入门教程博客系列,相反,读者真正想要理解 Paging 的使用,应该先尝试理解其分页组件的本质思想:

反思|Android 列表分页组件Paging的设计与实现:系统概述
反思|Android 列表分页组件Paging的设计与实现:架构设计与原理解析

以上两篇文章将对Paging分页组件进行了系统性的概述,笔者强烈建议 读者将以上两篇文章作为学习 Paging 阅读优先级 最高 的学习资料,所有其它的Paging中文博客阅读优先级都应该靠后

本文及相关引申阅读:

Android官方架构组件Paging:分页库的设计美学
Android官方架构组件Paging-Ex:为分页列表添加Header和Footer
Android官方架构组件Paging-Ex:列表状态的响应式管理

概述

PagingGoogle在2018年I/O大会上推出的适用于Android原生开发的分页库,如果您还不是很了解这个 官方钦定 的分页架构组件,欢迎参考笔者的这篇文章:

Android官方架构组件Paging:分页库的设计美学

笔者在实际项目中已经使用Paging半年有余,和市面上其它热门的分页库相比,Paging最大的亮点在于其 将列表分页加载的逻辑作为回调函数封装入 DataSource,开发者在配置完成后,无需控制分页的加载,列表会 自动加载 下一页数据并展示。

本文将阐述:为使用了Paging的列表添加HeaderFooter的整个过程、这个过程中遇到的一些阻碍、以及自己是如何解决这些阻碍的——如果您想直接浏览最终的解决方案,请直接翻阅本文的 最终的解决方案 小节。

初始思路

RecyclerView列表添加HeaderFooter并不是一个很麻烦的事,最简单粗暴的方式是将RecyclerViewHeader共同放入同一个ScrollView的子View中,但它无异于对RecyclerView自身的复用机制视而不见,因此这种解决方案并非首选。

更适用的解决方式是通过实现 多类型列表(MultiType),除了列表本身的Item类型之外,HeaderFooter也被视作一种Item,关于这种方式的实现网上已有很多文章讲解,本文不赘述。

在正式开始本文内容之前,我们先来看看最终的实现效果,我们为一个Student的分页列表添加了一个HeaderFooter

实现这种效果,笔者最初的思路也是通过 多类型列表 实现HeaderFooter,但是很快我们就遇到了第一个问题,那就是 我们并没有直接持有数据源

1.数据源问题

对于常规的多类型列表而言,我们可以轻易的持有List<ItemData>,从数据的控制而言,我更倾向于用一个代表Header或者Footer的占位符插入到数据列表的顶部或者底部,这样对于RecyclerView的渲染而言,它是这样的:

正如我所标注的,List<ItemData>中一个ItemData对应了一个ItemView——我认为为一个Header或者Footer单独创建对应一个Model类型是完全值得的,它极大增强了代码的可读性,而且对于复杂的Header而言,代表状态的Model类也更容易让开发者对其进行渲染。

这种实现方式简单、易读而不失优雅,但是在Paging中,这种思路一开始就被堵死了。

我们先看PagedListAdapter类的声明:

// T泛型代表数据源的类型,即本文中的 Student
public abstract class PagedListAdapter<T, VH extends RecyclerView.ViewHolder>extends RecyclerView.Adapter<VH> {// ...
}

因此,我们需要这样实现:

// 这里我们只能指定Student类型
class SimpleAdapter : PagedListAdapter<Student, RecyclerView.ViewHolder>(diffCallback) {// ...
}

有同学提出,我们可以将这里的Student指定为某个接口(比如定义一个ItemData接口),然后让StudentHeader对应的Model都去实现这个接口,然后这样:

class SimpleAdapter : PagedListAdapter<ItemData, RecyclerView.ViewHolder>(diffCallback) {// ...
}

看起来确实可行,但是我们忽略了一个问题,那就是本小节要阐述的:

我们并没有直接持有数据源

回到初衷,我们知道,Paging最大的亮点在于 自动分页加载,这是观察者模式的体现,配置完成后,我们并不关心 数据是如何被分页、何时被加载、如何被渲染 的,因此我们也不需要直接持有List<Student>(实际上也持有不了),更无从谈起手动为其添加HeaderItemFooterItem了。

以本文为例,实际上所有逻辑都交给了ViewModel

class CommonViewModel(app: Application) : AndroidViewModel(app) {private val dao = StudentDb.get(app).studentDao()fun getRefreshLiveData(): LiveData<PagedList<Student>> =LivePagedListBuilder(dao.getAllStudent(), PagedList.Config.Builder().setPageSize(15)                         //配置分页加载的数量.setInitialLoadSizeHint(40)              //初始化加载的数量.build()).build()
}

可以看到,我们并未直接持有List<Student>,因此list.add(headerItem)这种 持有并修改数据源 的方案几乎不可行(较真而言,其实是可行的,但是成本过高,本文不深入讨论)。

2.尝试直接实现列表

接下来我针对直接实现多类型列表进行尝试,我们先不讨论如何实现Footer,仅以Header而言,我们进行如下的实现:

class HeaderSimpleAdapter : PagedListAdapter<Student, RecyclerView.ViewHolder>(diffCallback) {// 1.根据position为item分配类型// 如果position = 1,视为Header// 如果position != 1,视为普通的Studentoverride fun getItemViewType(position: Int): Int {return when (position == 0) {true -> ITEM_TYPE_HEADERfalse -> super.getItemViewType(position)}}// 2.根据不同的viewType生成对应ViewHolderoverride fun onCreateViewHolder(parent: ViewGroup, viewType: Int): RecyclerView.ViewHolder {return when (viewType) {ITEM_TYPE_HEADER -> HeaderViewHolder(parent)else -> StudentViewHolder(parent)}}// 3.根据holder类型,进行对应的渲染override fun onBindViewHolder(holder: RecyclerView.ViewHolder, position: Int) {when (holder) {is HeaderViewHolder -> holder.renderHeader()is StudentViewHolder -> holder.renderStudent(getStudentItem(position))}}// 4.这里我们根据StudentItem的position,// 获取position-1位置的学生private fun getStudentItem(position: Int): Student? {return getItem(position - 1)}// 5.因为有Header,item数量要多一个override fun getItemCount(): Int {return super.getItemCount() + 1}// 省略其他代码...// 省略ViewHolder代码
}    

代码和注释已经将我的个人思想展示的很清楚了,我们固定一个Header在多类型列表的最上方,这也导致我们需要重写getItemCount()方法,并且在对Item进行渲染的onBindViewHolder()方法中,对Sutdent的获取进行额外的处理——因为多了一个Header,导致产生了数据源和列表的错位差—— 第n个数据被获取时,我们应该将其渲染在列表的第n+1个位置上

我简单绘制了一张图来描述这个过程,也许更加直观易懂:

代码写完后,直觉告诉我似乎没有什么问题,让我们来看看实际的运行效果:

Gif也许展示并不那么清晰,简单总结下,问题有两个:

  • 1.在我们进行下拉刷新时,因为Header更应该是一个静态独立的组件,但实际上它也是列表的一部分,因此白光一闪,除了Student列表,Header作为Item也进行了刷新,这与我们的预期不符;
  • 2.下拉刷新之后,列表 并未展示在最顶部,而是滑动到了一个奇怪的位置。

导致这两个问题的根本原因仍然是Paging计算列表的position时出现的问题:

对于问题1,Paging对于列表的刷新理解为 所有Item的刷新,因此同样作为ItemHeader也无法避免被刷新;

问题2依然也是这个问题导致的,在Paging获取到第一页数据时(假设第一页数据只有10条),Paging会命令更新position in 0..9Item,而实际上因为Header的关系,我们是期望它能够更新第position in 1..10Item,最终导致了刷新以及对新数据的展示出现了问题。

3.向Google和Github寻求答案

正如标题而言,我尝试求助于GoogleGithub,最终找到了这个链接:

PagingWithNetworkSample - PagedList RecyclerView scroll bug

如果您简单研究过PagedListAdapter的源码的话,您应该了解,PagedListAdapter内部定义了一个AsyncPagedListDiffer,用于对列表数据的加载和展示,PagedListAdapter更像是一个空壳,所有分页相关的逻辑实际都 委托 给了AsyncPagedListDiffer:

public abstract class PagedListAdapter<T, VH extends RecyclerView.ViewHolder>extends RecyclerView.Adapter<VH> {final AsyncPagedListDiffer<T> mDiffer;public void submitList(@Nullable PagedList<T> pagedList) {mDiffer.submitList(pagedList);}protected T getItem(int position) {return mDiffer.getItem(position);}public int getItemCount() {return mDiffer.getItemCount();}       public PagedList<T> getCurrentList() {return mDiffer.getCurrentList();}
}          

虽然Paging中数据的获取和展示我们是无法控制的,但我们可以尝试 瞒过 PagedListAdapter,即使Paging得到了position in 0..9List<Data>,但是我们让PagedListAdapter去更新position in 1..10的item不就可以了嘛?

因此在上方的Issue链接中,onlymash 同学提出了一个解决方案:

重写PagedListAdapter中被AsyncPagedListDiffer代理的所有方法,然后实例化一个新的AsyncPagedListDiffer,并让这个新的differ代理这些方法。

篇幅所限,我们只展示部分核心代码:

class PostAdapter: PagedListAdapter<Any, RecyclerView.ViewHolder>() {private val adapterCallback = AdapterListUpdateCallback(this)// 当第n个数据被获取,更新第n+1个positionprivate val listUpdateCallback = object : ListUpdateCallback {override fun onChanged(position: Int, count: Int, payload: Any?) {adapterCallback.onChanged(position + 1, count, payload)}override fun onMoved(fromPosition: Int, toPosition: Int) {adapterCallback.onMoved(fromPosition + 1, toPosition + 1)}override fun onInserted(position: Int, count: Int) {adapterCallback.onInserted(position + 1, count)}override fun onRemoved(position: Int, count: Int) {adapterCallback.onRemoved(position + 1, count)}}// 新建一个differprivate val differ = AsyncPagedListDiffer<Any>(listUpdateCallback,AsyncDifferConfig.Builder<Any>(POST_COMPARATOR).build())// 将所有方法重写,并委托给新的differ去处理override fun getItem(position: Int): Any? {return differ.getItem(position - 1)}// 将所有方法重写,并委托给新的differ去处理override fun submitList(pagedList: PagedList<Any>?) {differ.submitList(pagedList)}// 将所有方法重写,并委托给新的differ去处理override fun getCurrentList(): PagedList<Any>? {return differ.currentList}
}

现在我们成功实现了上文中我们的思路,一图胜千言:

4.另外一种实现方式

上一小节的实现方案是完全可行的,但我个人认为美中不足的是,这种方案 对既有的Adapter中代码改动过大

我新建了一个AdapterListUpdateCallback、一个ListUpdateCallback以及一个新的AsyncPagedListDiffer,并重写了太多的PagedListAdapter的方法——我添加了数十行分页相关的代码,但这些代码和正常的列表展示并没有直接的关系。

当然,我可以将这些逻辑都抽出来放在一个新的类里面,但我还是感觉我 好像是模仿并重写了一个新的PagedListAdapter类一样,那么是否还有其它的思路呢?

最终我找到了这篇文章:

Android RecyclerView + Paging Library 添加头部刷新会自动滚动的问题分析及解决

这篇文章中的作者通过细致分析Paging的源码,得出了一个更简单实现Header的方案,有兴趣的同学可以点进去查看,这里简单阐述其原理:

通过查看源码,以添加分页为例,Paging对拿到最新的数据后,对列表的更新实际是调用了RecyclerView.AdapternotifyItemRangeInserted()方法,而我们可以通过重写Adapter.registerAdapterDataObserver()方法,对数据更新的逻辑进行调整

// 1.新建一个 AdapterDataObserverProxy 类继承 RecyclerView.AdapterDataObserver
class AdapterDataObserverProxy extends RecyclerView.AdapterDataObserver {RecyclerView.AdapterDataObserver adapterDataObserver;int headerCount;public ArticleDataObserver(RecyclerView.AdapterDataObserver adapterDataObserver, int headerCount) {this.adapterDataObserver = adapterDataObserver;this.headerCount = headerCount;}@Overridepublic void onChanged() {adapterDataObserver.onChanged();}@Overridepublic void onItemRangeChanged(int positionStart, int itemCount) {adapterDataObserver.onItemRangeChanged(positionStart + headerCount, itemCount);}@Overridepublic void onItemRangeChanged(int positionStart, int itemCount, @Nullable Object payload) {adapterDataObserver.onItemRangeChanged(positionStart + headerCount, itemCount, payload);}// 当第n个数据被获取,更新第n+1个position@Overridepublic void onItemRangeInserted(int positionStart, int itemCount) {adapterDataObserver.onItemRangeInserted(positionStart + headerCount, itemCount);}@Overridepublic void onItemRangeRemoved(int positionStart, int itemCount) {adapterDataObserver.onItemRangeRemoved(positionStart + headerCount, itemCount);}@Overridepublic void onItemRangeMoved(int fromPosition, int toPosition, int itemCount) {super.onItemRangeMoved(fromPosition + headerCount, toPosition + headerCount, itemCount);}
}// 2.对于Adapter而言,仅需重写registerAdapterDataObserver()方法
//   然后用 AdapterDataObserverProxy 去做代理即可
class PostAdapter extends PagedListAdapter {@Overridepublic void registerAdapterDataObserver(@NonNull RecyclerView.AdapterDataObserver observer) {super.registerAdapterDataObserver(new AdapterDataObserverProxy(observer, getHeaderCount()));}
}

我们将额外的逻辑抽了出来作为一个新的类,思路和上一小节的十分相似,同样我们也得到了预期的结果。

经过对源码的追踪,从性能上来讲,这两种实现方式并没有什么不同,唯一的区别就是,前者是针对PagedListAdapter进行了重写,将Item更新的代码交给了AsyncPagedListDiffer;而这种方式中,AsyncPagedListDiffer内部对Item更新的逻辑,最终仍然是交给了RecyclerView.AdapternotifyItemRangeInserted()方法去执行的——两者本质上并无区别

5.最终的解决方案

虽然上文只阐述了Paging library如何实现Header,实际上对于Footer而言也是一样,因为Footer也可以被视为另外一种的Item;同时,因为Footer在列表底部,并不会影响position的更新,因此它更简单。

下面是完整的Adapter示例:

class HeaderProxyAdapter : PagedListAdapter<Student, RecyclerView.ViewHolder>(diffCallback) {override fun getItemViewType(position: Int): Int {return when (position) {0 -> ITEM_TYPE_HEADERitemCount - 1 -> ITEM_TYPE_FOOTERelse -> super.getItemViewType(position)}}override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): RecyclerView.ViewHolder {return when (viewType) {ITEM_TYPE_HEADER -> HeaderViewHolder(parent)ITEM_TYPE_FOOTER -> FooterViewHolder(parent)else -> StudentViewHolder(parent)}}override fun onBindViewHolder(holder: RecyclerView.ViewHolder, position: Int) {when (holder) {is HeaderViewHolder -> holder.bindsHeader()is FooterViewHolder -> holder.bindsFooter()is StudentViewHolder -> holder.bindTo(getStudentItem(position))}}private fun getStudentItem(position: Int): Student? {return getItem(position - 1)}override fun getItemCount(): Int {return super.getItemCount() + 2}override fun registerAdapterDataObserver(observer: RecyclerView.AdapterDataObserver) {super.registerAdapterDataObserver(AdapterDataObserverProxy(observer, 1))}companion object {private val diffCallback = object : DiffUtil.ItemCallback<Student>() {override fun areItemsTheSame(oldItem: Student, newItem: Student): Boolean =oldItem.id == newItem.idoverride fun areContentsTheSame(oldItem: Student, newItem: Student): Boolean =oldItem == newItem}private const val ITEM_TYPE_HEADER = 99private const val ITEM_TYPE_FOOTER = 100}
}

如果你想查看运行完整的demo,这里是本文sample的地址:

https://github.com/qingmei2/SamplePaging

6.更多优化点?

文末最终的方案是否有更多优化的空间呢?当然,在实际的项目中,对其进行简单的封装是更有意义的(比如Builder模式、封装一个HeaderFooter甚至两者都有的装饰器类、或者其它…)。

本文旨在描述Paging使用过程中 遇到的问题解决问题的过程,因此项目级别的封装和实现细节不作为本文的主要内容;关于HeaderFooterPaging中的实现方式,如果您有更好的解决方案,期待与您的共同探讨。

参考&感谢

  • Paging library 源码
  • Android RecyclerView + Paging Library 添加头部刷新会自动滚动的问题分析及解决
  • PagingWithNetworkSample - PagedList RecyclerView scroll bug

系列文章

争取打造 Android Jetpack 讲解的最好的博客系列

  • Android官方架构组件Lifecycle:生命周期组件详解&原理分析
  • Android官方架构组件ViewModel:从前世今生到追本溯源
  • Android官方架构组件LiveData: 观察者模式领域二三事
  • Android官方架构组件Paging:分页库的设计美学
  • Android官方架构组件Paging-Ex:为分页列表添加Header和Footer
  • Android官方架构组件Paging-Ex:列表状态的响应式管理
  • Android官方架构组件Navigation:大巧不工的Fragment管理框架
  • Android官方架构组件DataBinding-Ex:双向绑定篇

Android Jetpack 实战篇

  • 开源项目:MVVM+Jetpack实现的Github客户端
  • 开源项目:基于MVVM, MVI+Jetpack实现的Github客户端
  • 总结:使用MVVM尝试开发Github客户端及对编程的一些思考

关于我

Hello,我是却把清梅嗅,如果您觉得文章对您有价值,欢迎 ❤️,也欢迎关注我的个人博客或者Github。

如果您觉得文章还差了那么点东西,也请通过关注督促我写出更好的文章——万一哪天我进步了呢?

  • 我的Android学习体系
  • 关于文章纠错
  • 关于知识付费

这篇关于Android官方架构组件Paging-Ex:为分页列表添加Header和Footer的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/397188

相关文章

Android数据库Room的实际使用过程总结

《Android数据库Room的实际使用过程总结》这篇文章主要给大家介绍了关于Android数据库Room的实际使用过程,详细介绍了如何创建实体类、数据访问对象(DAO)和数据库抽象类,需要的朋友可以... 目录前言一、Room的基本使用1.项目配置2.创建实体类(Entity)3.创建数据访问对象(DAO

Python中列表的高级索引技巧分享

《Python中列表的高级索引技巧分享》列表是Python中最常用的数据结构之一,它允许你存储多个元素,并且可以通过索引来访问这些元素,本文将带你深入了解Python列表的高级索引技巧,希望对... 目录1.基本索引2.切片3.负数索引切片4.步长5.多维列表6.列表解析7.切片赋值8.删除元素9.反转列表

Android WebView的加载超时处理方案

《AndroidWebView的加载超时处理方案》在Android开发中,WebView是一个常用的组件,用于在应用中嵌入网页,然而,当网络状况不佳或页面加载过慢时,用户可能会遇到加载超时的问题,本... 目录引言一、WebView加载超时的原因二、加载超时处理方案1. 使用Handler和Timer进行超

JS常用组件收集

收集了一些平时遇到的前端比较优秀的组件,方便以后开发的时候查找!!! 函数工具: Lodash 页面固定: stickUp、jQuery.Pin 轮播: unslider、swiper 开关: switch 复选框: icheck 气泡: grumble 隐藏元素: Headroom

mybatis的整体架构

mybatis的整体架构分为三层: 1.基础支持层 该层包括:数据源模块、事务管理模块、缓存模块、Binding模块、反射模块、类型转换模块、日志模块、资源加载模块、解析器模块 2.核心处理层 该层包括:配置解析、参数映射、SQL解析、SQL执行、结果集映射、插件 3.接口层 该层包括:SqlSession 基础支持层 该层保护mybatis的基础模块,它们为核心处理层提供了良好的支撑。

百度/小米/滴滴/京东,中台架构比较

小米中台建设实践 01 小米的三大中台建设:业务+数据+技术 业务中台--从业务说起 在中台建设中,需要规范化的服务接口、一致整合化的数据、容器化的技术组件以及弹性的基础设施。并结合业务情况,判定是否真的需要中台。 小米参考了业界优秀的案例包括移动中台、数据中台、业务中台、技术中台等,再结合其业务发展历程及业务现状,整理了中台架构的核心方法论,一是企业如何共享服务,二是如何为业务提供便利。

如何在页面调用utility bar并传递参数至lwc组件

1.在app的utility item中添加lwc组件: 2.调用utility bar api的方式有两种: 方法一,通过lwc调用: import {LightningElement,api ,wire } from 'lwc';import { publish, MessageContext } from 'lightning/messageService';import Ca

活用c4d官方开发文档查询代码

当你问AI助手比如豆包,如何用python禁止掉xpresso标签时候,它会提示到 这时候要用到两个东西。https://developers.maxon.net/论坛搜索和开发文档 比如这里我就在官方找到正确的id描述 然后我就把参数标签换过来

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

Android平台播放RTSP流的几种方案探究(VLC VS ExoPlayer VS SmartPlayer)

技术背景 好多开发者需要遴选Android平台RTSP直播播放器的时候,不知道如何选的好,本文针对常用的方案,做个大概的说明: 1. 使用VLC for Android VLC Media Player(VLC多媒体播放器),最初命名为VideoLAN客户端,是VideoLAN品牌产品,是VideoLAN计划的多媒体播放器。它支持众多音频与视频解码器及文件格式,并支持DVD影音光盘,VCD影