Kotlin Jetpack Paging3 和Flow结合使用注意点

本文首发于公众号“AntDream”,欢迎微信搜索“AntDream”或扫描文章底部二维码关注,和我一起每天进步一点点

在用DataBinding时要注意DataBinding访问的是静态方法

在kotlin中就要用companion object和@JvmStatic

class ImageViewBindingAdapter {//里面的BindingAdapter方法必须是静态方法,否则会编译会报错//DataBinding调用必须是静态方法companion object {@JvmStatic@BindingAdapter("image")fun setImage(imageView:ImageView, url: String){if (!TextUtils.isEmpty(url)){//加载网络图片}else{imageView.setBackgroundColor(Color.GRAY)}}}
}

paging3分页数据错乱的问题

在计算paging的prevKey和nextKey,也就是上一页,下一页的时候,需要考虑PagingConfig中的initialLoadSize参数

fun loadMovie(): Flow<PagingData<Movie>> {return Pager(config = PagingConfig(pageSize = 8,//第一次加载的数量,16也就是2页,默认是3*pageSizeinitialLoadSize = 16),pagingSourceFactory = {MoviePagingSource()}).flow
}class MoviePagingSource: PagingSource<Int, Movie>() {override fun getRefreshKey(state: PagingState<Int, Movie>): Int? {//第一次加载return 1}override suspend fun load(params: LoadParams<Int>): LoadResult<Int, Movie> {val currentPage = params.key ?: 1val pageSize = params.loadSizeval movies = RetrofitClient.createApi(MoviesApi::class.java).getMovies(currentPage, pageSize)var prevKey: Int? = nullvar nextKey: Int? = null//PagingConfig中的2个参数val realPageSize = 8val initialLoadSize = 16if (currentPage == 1){prevKey = nullnextKey = initialLoadSize / realPageSize + 1}else{prevKey = currentPage - 1nextKey = if (movies.hasMore) currentPage + 1 else null}//下面这样计算,在initialLoadSize不等于realPageSize的时候会有数据错乱//prevKey = if (currentPage == 1) null else currentPage - 1//nextKey = if (movies.hasMore) currentPage + 1 else nullreturn try {LoadResult.Page(data = movies.movieList,prevKey = prevKey,nextKey = nextKey)}catch (e: Exception){e.printStackTrace()return LoadResult.Error(e)}}}

给paging加上拉加载更多

recycleView.adapter = movieAdapter.withLoadStateFooter(MovieLoadMoreAdapter(this@MainActivity))

只需要加个适配器就可以

class MovieLoadMoreAdapter(private val context: Context): LoadStateAdapter<BindViewHolder>() {override fun onBindViewHolder(holder: BindViewHolder, loadState: LoadState) {}override fun onCreateViewHolder(parent: ViewGroup, loadState: LoadState): BindViewHolder {val binding = MovieLoadmoreBinding.inflate(LayoutInflater.from(context), null, false)return BindViewHolder(binding)}
}

paging加上下拉刷新

  • 布局加上SwipeRefreshLayout
<androidx.swiperefreshlayout.widget.SwipeRefreshLayoutandroid\:id="@+id/swipeRefreshLayout"android\:layout\_width="match\_parent"android\:layout\_height="match\_parent"><androidx.recyclerview.widget.RecyclerViewandroid:id="@+id/recycleView"android:layout_width="match_parent"android:layout_height="match_parent"app:layoutManager="androidx.recyclerview.widget.LinearLayoutManager" /></androidx.swiperefreshlayout.widget.SwipeRefreshLayout>
  • 加上刷新监听
swipeRefreshLayout.setOnRefreshListener {movieAdapter.refresh()
}
  • 监听刷新结束
lifecycleScope.launchWhenCreated {movieAdapter.loadStateFlow.collectLatest { state ->mBinding.swipeRefreshLayout.isRefreshing = state.refresh is LoadState.Loading}
}

下拉刷新后,底部上拉加载更多的loadmore的动画不显示

  • PageConfig还有一个属性是prefetchDistance,预刷新的距离,距离最后一个item多远时加载数据,默认为pageSize
  • 当prefetchDistance很小,并且initialLoadSize也很小时,就会出现上面的bug。比如initialLoadSize=8,prefetchDistance=1时
  • 解决办法也比较简单,2个属性设置的大一点就行了

APP横竖屏切换之后paging加载的数据没有缓存起来

  • ViewModel缓存数据要在属性中
  • 还有就是paging返回的是flow,需要用**cachedIn(viewModelScope)**来让paging的flow的生命周期和ViewModelScope的生命周期保持一致,也就是和activity保持一致
class MovieViewModel: ViewModel() {private val movies by lazy {Pager(config = PagingConfig(pageSize = 8,//第一次加载的数量,16也就是2页initialLoadSize = 16,//预刷新的距离,距离最后一个item多远时加载数据,默认为pageSizeprefetchDistance = 8),pagingSourceFactory = {MoviePagingSource()}).flow.cachedIn(viewModelScope)}fun loadMovie(): Flow<PagingData<Movie>> = movies}

欢迎关注我的公众号查看更多精彩文章!

AntDream

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/21532.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

MySQl基础入门⑬.5

创建多表连接查询 表准备 CREATE TABLE 员工信息 (员工号 INT(11) NOT NULL AUTO_INCREMENT PRIMARY KEY,姓名 VARCHAR(50) NOT NULL,性别 ENUM(男, 女) NOT NULL,出生日期 DATE NOT NULL,部门 VARCHAR(50) NOT NULL,手机号码 VARCHAR(20) NOT NULL,-- 根据数据库不同&#x…

NVIDIA NIM推理微服务集成6款新模型

NIM(NVIDIA Inference Microservice)最新模型实例 得益于不断涌现的突破性基础模型&#xff0c;生成式 AI 正在彻底改变各行各业的几乎所有用例。这些模型能够理解背景和原因&#xff0c;从而生成高质量的内容和高精度答案。 NVIDIA 不断优化和发布新的模型。本文将为您介绍最…

Python入门实践:从基础到应用的全方位探索

Python入门实践&#xff1a;从基础到应用的全方位探索 在数字化浪潮席卷全球的今天&#xff0c;掌握一门编程语言已成为一项重要的技能。而Python&#xff0c;作为一门简洁、易读且功能强大的编程语言&#xff0c;受到了越来越多人的青睐。本文将从四个方面、五个方面、六个方…

ctfshow-web入门-爆破(web25)及php_mt_seed工具的安装与使用

爆个&#x1f528;&#xff0c;不爆了 hexdec() 函数用于将十六进制字符串转换为十进制数&#xff1b; 注意&#xff1a; 我最开始做这道题时看错了&#xff0c;误以为随机数的种子直接来自于 flag 的前八位&#xff0c;以为就是 ctfshow{ 这八个字符然后 md5 加密再截取&a…

【DSP】【第六篇】开发支持包

1. SYS/BIOS发展历史 2. 下载链接 全目录链接。 2.1 SYS/BIOS DSP/BIOS和SYS/BIOS链接 2.1.1 DSP/BIOS 2.1.2 SYS/BIOS SYS/BIOS下载链接下图红框标记的是CCSv5可以使用的最新的版本。 2.1.2.1 网络驱动与协议栈 SYS/BIOS组件之网络驱动与协议栈 &#xff08;1&#xf…

yolov8改进之嵌入Gold层

# 加载模型 model = YOLO("yolov8n.yaml") # 从头开始构建新模型 model = YOLO("yolov8n.pt") # 加载预训练模型(建议用于训练) # 使用模型 model.train(data="coco128.yaml", epochs=3) # 训练模型 metrics = model.val() # 在验证集上评…

【面试题-003】重载和重写的区别

在面向对象编程中&#xff0c;重载&#xff08;Overloading&#xff09;和重写&#xff08;Overriding&#xff09;是两个不同的概念&#xff0c;它们用于实现多态性&#xff1a; 重载&#xff08;Overloading&#xff09;&#xff1a; 重载发生在同一个类中&#xff0c;当多个…

2024年Google算法更新打击低质量(如AI生成)内容后,英文SEO优化人员该如何调整谷歌SEO优化策略?

3月5日&#xff0c;谷歌发布了2024年的首次算法更新。与以往更新不同&#xff0c;本次更新更加复杂&#xff0c;这次更新旨在提高搜索结果的质量和相关性&#xff0c;可能对外贸网站排名和流量产生显著影响。也将产生更大的网站数据波动。但在担心自己的网站数据受到影响之前&a…

MYSQL三、MYSQL的函数与约束

一、函数 函数&#xff1a;是指一段可以直接被另一段程序调用的程序或代码。MySQL中的函数主要分为以下四类&#xff1a; 字符串函数、数值函数、日期函数、流程函数。 1、字符串函数 -- concat : 字符串拼接 select concat(Hello , MySQL); -- 输出&#xff1a; Hello M…

通用代码生成器应用场景六,为完善的应用系统收集需求

通用代码生成器应用场景六&#xff0c;为完善的应用系统收集需求 使用急就章功能可以开发一个简单的应用先凑和着使用。此应用系统也可以成为完善的应用系统的原型和祖先。如果您新规划一个完善的应用系统&#xff0c;您可以先使用通用代码生成器生成一个临时使用的系统&#x…

【TB作品】MSP430F5529,单片机,Picture to pixels,乌鸦喝水OLED

功能 Picture to pixels. Use bitmaps to tell a story. Convert pictures to bitmaps and store the bitmaps in a header file. In the main program, draw the pictures on the OLED screen in sequence to tell a story. Use the text to help tell the story 乌鸦喝水故…

Python一般用什么IDE:深入剖析四大主流选择

Python一般用什么IDE&#xff1a;深入剖析四大主流选择 在Python编程的世界里&#xff0c;选择合适的集成开发环境&#xff08;IDE&#xff09;对于提升编程效率和体验至关重要。本文将从四个方面、五个方面、六个方面和七个方面&#xff0c;深入剖析Python开发者常用的四大主…

【linux】(1)文件操作及vi

文件和目录的创建 创建文件 touch 命令&#xff1a;创建一个新的空文件。 touch filename.txtecho 命令&#xff1a;创建一个文件并写入内容。 echo "Hello, World!" > filename.txtcat 命令&#xff1a;将内容写入文件。 cat > filename.txt然后输入内容&…

人脸识别系统之动态人脸识别

二&#xff0e;动态人脸识别 1.摄像头人脸识别 1.1.导入资源包 import dlib import cv2 import face_recognition from PIL import Image, ImageTk import tkinter as tk import os注&#xff1a;这些导入语句允许您在代码中使用这些库和模块提供的功能&#xff0c;例如创建…

Cacti EZ中文版 12.2.27 ISO 下载安装

简介 修改了yum源为中国高校联合镜像源 github改为gitee。 系统增加中文语言包。 修改时区为东八区。 增加了常用的软件包。 PS&#xff1a;CactiEZ是一个自动化安装cacti和插件的ISO镜像&#xff0c;本教程的ISO是基于官方的IOS针对国内网络做了修改。 可按照目前最新的Ca…

方法递归案例

文件搜索 需求&#xff1a;从D盘中,搜索idea64.exe这个文件,找到后输出其位置 public static void main(String[] args) {searchFile(new File("D:/") , "idea64.exe");}/*** 去目录搜索某个文件* param dir 目录* param fileName 要搜索的文件名称*/pub…

Java1.8 vue版家政服务系统成品源码 家政管家系统源码 家政月嫂系统源码 家政保洁系统源码 在线派单,师傅入驻全套商业源码

Java1.8 vue版家政服务系统成品源码 家政管家系统源码 家政月嫂系统源码 家政保洁系统源码 在线派单&#xff0c;师傅入驻全套商业源码 一、系统定义 家政上门服务系统是一种利用互联网技术&#xff0c;将家政服务需求与专业的家政服务人员进行高效匹配的平台。它允许用户通过…

【OceanBase诊断调优】—— obdiag 工具助力OceanBase数据库诊断调优(DBA 从入门到实践第八期)

1. 前言 昨天给大家分享了【DBA从入门到实践】第八期&#xff1a;OceanBase数据库诊断调优、认证体系和用户实践 中obdiag的部分&#xff0c;今天将其中的内容以博客的形式给大家展开一下&#xff0c;方便大家阅读。 2. 正文 在介绍敏捷诊断工具之前&#xff0c;先说说OceanBa…

springboot配置

springboot配置 配置文件分类 springboot提供了多种属性配置方式 application.xml(只有老的spring项目使用)application.propertiesapplication.yml(或yaml) 优先级:properties>yml>yaml 配置文件 pom.xml★ 在Spring Boot项目中&#xff0c;pom.xml文件是Maven项目…

x264帧级码率控制解析

目前x264代码中表面看到的码率控制方式有CQP,CRF和ABR三种。 CQP(Constant Quantitative Parameters), 固定QP,直接整帧都用相同的QP值。CRF (Constant Rate Factor),固定码率因子,更注重质量,x264默认的码控方式,CRF会动态的调整每帧的QP,对比CQP在同等码率下CRF能…