[Android+JetPack] (Java实现) Retrofit2+RxJava3+Paging3+RecyclerView 实现加载网络数据例子 记录

news2025/1/19 14:19:08

文章目录

  • 前言
  • 参考链接
  • 依赖库及版本
  • Demo效果
  • 接口及数据展示
  • 各项模块
    • `Retrofit2`
      • Bean,对应上面的接口返回.
      • Service API部分
    • `Paging3`
      • `PagingSource`以及 `RxPagingSource`
      • `PagingDataAdapter` 适配器
      • `ViewModel`
      • `PublicInfoPage /Activity`
  • 最后

前言


继续安卓学习之旅,本章的主要目标是:

1.完成一个无限上拉加载的列表(Paging3 + RecyclerView)
2.加载的是网络数据, 要采用主流的 Retrofit+okhttp方式
3.在了解了RxJava之后,也希望用上Rxjava
4.用到ViewModel来配合,以及一些jetpack的东西都用上

(为什么不用Paging2? 这里主要是看说3比2还要方便些,所以就偷懒没去用Paging2)

参考链接

这些是在学习和搜索中看到的比较好的文章,不过他们要么是kotlin 要么是RxJava2,都不是能直接套上去就用的,但是从文章里面总结归纳,也是有借鉴效果的.

  1. SmartRefreshLayout-github 这个后期再结合Paging3,完成一个有酷炫下拉及淘宝二楼效果的的demo
  2. Java实现)使用官方Paging3分页库实现RecyclerView加载更多(loadmore)的功能 这个较为简洁,没那么多原理的描述,方便更实战的理解借鉴
  3. Android paging3 使用和踩坑经验分享 这个虽然是kotlin,不过里面一些名词的解释不错, 适合快速扫盲
  4. Jetpack新成员,Paging3从吐槽到真香

依赖库及版本

为什么要说这个, 因为在实际百度各方面资料的时候,没仔细区分好版本,导致在练习过程中走了不少弯路,踩了坑.为避免这个情况,这里列出本Demo中的各个依赖库及版本

Retrofix2

    // 引入 retrofix 网络框架(自带okhttp)
    // github :https://github.com/square/retrofit
    // 视频教学
    // https://www.bilibili.com/video/BV1vV411W75V?p=4&vd_source=3dc64571e08f84008d5c43796c009480
    implementation "com.squareup.retrofit2:retrofit:2.9.0"
    implementation 'com.squareup.retrofit2:converter-gson:2.9.0'
    implementation 'com.squareup.retrofit2:adapter-rxjava3:2.9.0'

Rxjava3

    // 支持RxJava/RxAndroid
    implementation 'io.reactivex.rxjava3:rxandroid:3.0.2'
    implementation 'io.reactivex.rxjava3:rxjava:3.1.5'

Paging3

    // 引入 paging 3
    // 注意, 由于上面我们用的是 retrofit2 + rxjava3
    // 所以,在使用 paging3的时候, 要选 支持rxjava3的 paging-rxjava3
    // 切记版本对应好
    def paging_version = "3.1.1"
    implementation "androidx.paging:paging-runtime:$paging_version"
//    // optional - RxJava2 support
//    implementation "androidx.paging:paging-rxjava2:$paging_version"
    // optional - RxJava3 support
    implementation "androidx.paging:paging-rxjava3:$paging_version"

这里稍微提一下, 如果用的是RxJava3, 就使用RxJava3 support的可选项, 不然不匹配,但同时也造成另一个问题, 这里插入说一下哈

就是
包括目前官网(点击进入)那边的, 关于对RxPagingSource的示例里面, 也应该用的还是RxJava2,如果你和我一样用RxJava3,那大概率在做map的时候,会报错说, 类型转换失败, 不能用 this::toLoadResult
这个稍后再说…

在这里插入图片描述

Demo效果

请添加图片描述

一个简单的demo

接口及数据展示

请添加图片描述

各项模块

Retrofit2

Bean,对应上面的接口返回.

Response_public_info_bean

package retrofit.bean;

import java.util.List;

/**
 * @author: tiannan
 * @time: 2023/4/12.
 * @email: tianNanYiHao@163.com
 * @descripetion: 此处添加描述
 */
public class Response_public_info_bean {
    private String msg;
    private String code;
    private Datas data;


    public String getMsg() {
        return msg;
    }

    public void setMsg(String msg) {
        this.msg = msg;
    }

    public String getCode() {
        return code;
    }

    public void setCode(String code) {
        this.code = code;
    }

    public Datas getData() {
        return data;
    }

    public void setData(Datas data) {
        this.data = data;
    }


    public class Datas {
        private int pageSize;
        private List<Cell> list;

        public int getPageSize() {
            return pageSize;
        }

        public void setPageSize(int pageSize) {
            this.pageSize = pageSize;
        }

        public List<Cell> getList() {
            return list;
        }

        public void setList(List<Cell> list) {
            this.list = list;
        }

        @Override
        public String toString() {
            return "Datas{" +
                    "list=" + list +
                    '}';
        }

        public class Cell {
            private String productName;
            private String productTypeName;
            private String riskRateName;

            private int id;

            private int pageNum; // 增加两个下标 page页下标
            private int indexNum;// 增加两个下标 newsInfo(cell)页下标

            @Override
            public String toString() {
                return "News{" +
                        "productName='" + productName + '\'' +
                        ", productTypeName='" + productTypeName + '\'' +
                        ", riskRateName='" + riskRateName + '\'' +
                        ", id=" + id +
                        ", pageNum=" + pageNum +
                        ", indexNum=" + indexNum +
                        '}';
            }

            public String getProductName() {
                return productName;
            }

            public void setProductName(String productName) {
                this.productName = productName;
            }

            public String getProductTypeName() {
                return productTypeName;
            }

            public void setProductTypeName(String productTypeName) {
                this.productTypeName = productTypeName;
            }

            public String getRiskRateName() {
                return riskRateName;
            }

            public void setRiskRateName(String riskRateName) {
                this.riskRateName = riskRateName;
            }

            public int getId() {
                return id;
            }

            public void setId(int id) {
                this.id = id;
            }

            public int getPageNum() {
                return pageNum;
            }

            public void setPageNum(int pageNum) {
                this.pageNum = pageNum;
            }

            public int getIndexNum() {
                return indexNum;
            }

            public void setIndexNum(int indexNum) {
                this.indexNum = indexNum;
            }
        }


    }
}

Service API部分

这里不展开说太多 Retrofit2部分的东西,这里只贴一下和本章有关的部分代码

    /**
     * 请求公募数据列表
     * @param map
     * @return
     */
    @GET(App_Url.admin_getPublicProductInfoPageList)
    Single<Response_public_info_bean> admin_getPublicProductInfoPageList(@QueryMap HashMap<String,String> map);

以上部分完成后 ,能够通过RxJava3 + Retrofit2 配合完成一次网络请求, 基本就算完成Demo一半功能了

Paging3

PagingSource以及 RxPagingSource

这里一开始我用错了RxPagingSource的导入版本, 用成了rxjava2的,踩了些坑
在这里插入图片描述
当时导入了paging.rxjava2这个版本,
主要原因还是在配置依赖的时候, 把paging-rxjava2:3.1.1版本也同步了
在这里插入图片描述
版本问题注意好

回到 RxPagingSource
按照网上的文章的示例, 先处理 loadSingle()函数的实现,这里有坑就是上面说的, RxJava3 + Paging3的情况下,
在这里插入图片描述
return网络请求的时候,会报错
在这里插入图片描述
我这里没有在详细探究是否由于RxJava2的原因导致不能用 ::这种双冒号的写法
这里仅仅贴一下RxJava2下的map 和 RxJava3下的map的源码区别
RxJava2版本:
在这里插入图片描述
RxJava3版本:
在这里插入图片描述
确实有一点区别, 这个先放一放, 等后期有空再看怎么处理…

先直接看怎么去写这个 this::toLoadResult
首先,既然通过Retrofit2,我们已经定义了网络请求的返回值
在这里插入图片描述
那么我们在RxPagingSourceloadSingle()中, 会去调用网络请求,得到一个 Single<Response_public_info_bean>
在这里插入图片描述
我们再看官网例子的这部分代码
在这里插入图片描述
注意看返回值其实是LoadResult<Integer, User> ,或者说,在本文章 我们要的返回值其实是 LoadResult<Integer, Response_public_info_bean.Datas.Cell>
所以.对于map操作符,在Rxjava3的下, 我们是可以自己去提供一个Function<T,R>,这里面T就是我们上面的Response_public_info_bean
R就是Response_public_info_bean.Datas.Cell>

所以代码就是

在这里插入图片描述
(这里要注意下prevkey, 和 nextKey的入参 , 要做好逻辑判断, 一开始我参考别人的代码, 在prevKey填的是null, 在nextKey填入的是nextPageKey+1,结果导致加载页码瞬间冲到了几百页, 其实总page数量才不过十几页)

为了更加清晰明了的展示页码和条数下标, 我又添加了一个map操作符, 是给cell这个Bean数据再添加一下当前所属的页码 和 当前的下标
Function的入参依然是 Response_public_info_bean 返回也还是 Response_public_info_bean, 相当于我们就对Response_public_info_bean数据做了个数据加工
在这里插入图片描述

所以,基于RxJava3的map操作符这边就可以这样返回
在这里插入图片描述

PagingDataAdapter 适配器

public class PublicInfoAdapter extends PagingDataAdapter<Response_public_info_bean.Datas.Cell, PublicInfoAdapter.Holder> {


    public PublicInfoAdapter(@NotNull DiffUtil.ItemCallback<Response_public_info_bean.Datas.Cell> diffCallback) {
        super(diffCallback);
    }


    @NonNull
    @NotNull
    @Override
    public Holder onCreateViewHolder(@NonNull @NotNull ViewGroup parent, int viewType) {
        
        PublicInfoCellBinding publicInfoCellBinding = PublicInfoCellBinding.inflate(LayoutInflater.from(parent.getContext()), parent, false);
        return new Holder(publicInfoCellBinding);
    }



    @Override
    public void onBindViewHolder(@NonNull @NotNull Holder holder, int position) {
        Response_public_info_bean.Datas.Cell cell = getItem(position);
        Log.d("dfaddfsa", "onBindViewHolder: " + cell.getPageNum() +"."+ cell.getIndexNum() + "---" + cell.getProductName());

        holder.publicInfoCellBinding.textTitle.setText(cell.getProductName());
        holder.publicInfoCellBinding.underflag.setText("第"+cell.getPageNum() + "页.第" + cell.getIndexNum()+"条");
    }

    public class Holder extends RecyclerView.ViewHolder {

        /**
         * 给每个 Holder 实例 一个 viewBinding
         */
        private PublicInfoCellBinding publicInfoCellBinding;

        public Holder(@NonNull @NotNull View itemView) {
            super(itemView);
        }

        public Holder(@NonNull PublicInfoCellBinding publicInfoCellBinding){
            super(publicInfoCellBinding.getRoot());
            this.publicInfoCellBinding = publicInfoCellBinding;
        }
    }
}

这里没太多可以说的,网上基本讲明白了, 我仅仅分享我遇到的一个问题
页面列表在加载完成后, 出现了一屏数据的重复渲染, 而且随着页面的滚动,该组数据的最后一条一直在渲染不同内容(但随着日志打印,数据都是正常输出的)
最后通过UI观察, 感觉最后一条数据随滚动而渲染,有点想是for循环没拦住的那种意思
就猜想,是不是 PagingDataAdapter 里面没处理好,
后来果然发现, PublicInfoCellBinding publicInfoCellBinding;一不小心写成了全局的,而不是给每个Holder一个PublicInfoCellBinding publicInfoCellBinding;, 最后修复下即可

这里还是由于对PagingDataAdapter的不够熟悉, 刚写着玩意儿,才出现的低级错误

ViewModel

vm部分,网上也大同小异 ,写demo过程中未出现过多的坎儿

public class PublicInfoViewModel extends ViewModel {

    // paging3 page对象
    Pager<Integer, Response_public_info_bean.Datas.Cell> pager;
    // paging3 数据源对象
    PublicInfoSource publicInfoSource;
    // rxjava3 的 obserable 可观察对象
    Flowable<PagingData<Response_public_info_bean.Datas.Cell>> pagingDataFlowable;

    public PublicInfoViewModel(Context context) {
        CoroutineScope viewModelScope = ViewModelKt.getViewModelScope(this);

        publicInfoSource = new PublicInfoSource();

        // Maximum size must be at least pageSize + 2*prefetchDist, pageSize=20, prefetchDist=20, maxSize=20
        /**
         * pageSize 每页多少个条目
         * prefetchDistance 预加载下一页的距离,滑动到倒数第几个条目就加载下一页,无缝加载(可选)默认值是pageSize
         * enablePlaceholders 是否启用条目占位,当条目总数量确定的时候;列表一次性展示所有条目,
         *                    但是没有数据;在adapter的onBindViewHolder里面绑定数据时候,是空数据,判断是空数据展示对应的占位item;可选,默认开启
         * initialLoadSize 第一页加载条目数量 ,可选,默认值是 3*pageSize (有时候需要第一页多点数据可用)
         * maxSize : 定义列表最大数量;可选,默认值是:Int.MAX_VALUE
         * jumpThreshold : 暂时还不知道用法,从文档注释上看,是滚动大距离导致加载失效的阈值;可选,默认值是:Int.MIN_VALUE (表示禁用此功能)
         *
         */
        PagingConfig pagingConfig = new PagingConfig(20,1,false,20*3);

        pager = new Pager<Integer, Response_public_info_bean.Datas.Cell>(pagingConfig, () -> publicInfoSource);

        pagingDataFlowable = PagingRx.getFlowable(pager);

        PagingRx.cachedIn(pagingDataFlowable, viewModelScope);
    }


    public Flowable<PagingData<Response_public_info_bean.Datas.Cell>> getPagingDataFlowable() {
        return pagingDataFlowable;
    }
}

PublicInfoPage /Activity

这里要注意的是, setLayoutManager要设置 否则啥也不展示


public class PublicInfoPage extends AppCompatActivity {

    ActivityNewsPageBinding newsPageBinding;

    PublicInfoViewModel newsViewModel;
    PublicInfoAdapter newsAdapter;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        newsPageBinding = ActivityNewsPageBinding.inflate(getLayoutInflater());


        newsAdapter = new PublicInfoAdapter(new DiffUtil.ItemCallback<Response_public_info_bean.Datas.Cell>() {
            @Override
            public boolean areItemsTheSame(@NonNull @NotNull Response_public_info_bean.Datas.Cell oldItem, @NonNull @NotNull Response_public_info_bean.Datas.Cell newItem) {
                return oldItem.getId() == newItem.getId();
            }

            @Override
            public boolean areContentsTheSame(@NonNull @NotNull Response_public_info_bean.Datas.Cell oldItem, @NonNull @NotNull Response_public_info_bean.Datas.Cell newItem) {
                return oldItem.getProductName().equals(newItem.getProductName()) && oldItem.getProductTypeName().equals(newItem.getProductTypeName());
            }
        });
        newsPageBinding.recicleView.setAdapter(newsAdapter);
        newsPageBinding.recicleView.setLayoutManager(new LinearLayoutManager(this));

        setContentView(newsPageBinding.getRoot());
    }


    @Override
    protected void onResume() {
        super.onResume();

        newsViewModel = new PublicInfoViewModel(this);
        newsViewModel.getPagingDataFlowable().subscribe(new Consumer<PagingData<Response_public_info_bean.Datas.Cell>>() {
            @Override
            public void accept(PagingData<Response_public_info_bean.Datas.Cell> newsPagingData) throws Throwable {
                newsAdapter.submitData(getLifecycle(), newsPagingData);
            }
        });

    }
}

最后

以上就是这样了.
SmartRefreshLayout 也可以结合Paging3
这个有空也看一下,
安卓的玩法确实和iOS不一样, 也和RN不一样, 但互相又看得到对方的影子

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

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

相关文章

2023年mathorcup杯A题代码小技巧总结

写篇杂文&#xff0c;不属于我的任何一个栏目&#xff0c;没啥案例含义&#xff0c;主要是记录一下mathorcup杯期间写的代码技巧。 mathorcup杯大部分都是组合优化问题&#xff0c;这让我擅长的机器学习都深度学习毫无作用....... 还好这个A题是可以遍历求所有解的&#xff0c…

SAP MDG —— 使用DIF导入物料主数据 Part4 开发篇

文章目录 通过CSV文件创建物料主数据的一些建议在DIF中使用CSV文件文件转换器类型BAdIBAdI 示例代码测试 DIF全篇总结 通过CSV文件创建物料主数据的一些建议 在Staging Area中创建/修改小批量的物料主数据&#xff0c;推荐使用标准文件上载功能&#xff08;USMD_FILE_UPLOAD&am…

计算机基础--Redis

参考文献 https://zhuanlan.zhihu.com/p/599663544 基础知识 1. Redis的数据类型及使用场景 Redis支持五种数据类型&#xff1a;string&#xff08;字符串&#xff09;&#xff0c;hash&#xff08;哈希&#xff09;&#xff0c;list&#xff08;列表&#xff09;&#xff…

使用assembly插件对boot项目打zip和war部署包

目录 一、maven-assembly-plugin插件的简单使用 1、什么是assembly&#xff1f; 2. 常见的maven插件 3、如何使用&#xff1f; 二、如何通过assembly打不同的包 三、boot项目如何转成war包部署 背景&#xff1a;之前项目上已经使用了assembly对多个boot项目分别打zip包且…

MATLAB应用笔记

其他 1、NaN值 MATLAB判断数据是否为NaN可以直接使用函数&#xff1a;isnan() 三、数据分析 1、相关性 均值、方差、协方差、标准差、相关系数 mean() %均值 nanmean()%去除NAN值求均值 var() %方差 cov() %协方差 std() %标准差 corrcoef(B,b) %R 相关系数plot()…

[ 云原生 | Docker ] 构建高可用性的 SQL Server:Docker 容器下的主从同步实现指南

文章目录 一、前言二、SQL Server 主从同步的原理介绍三、具体的搭建过程3.1 准备工作3.1.1 卸载旧版本&#xff08;如果有&#xff0c;可选&#xff0c;非必须&#xff09;3.1.2 安装 Docker3.1.3 验证本地 Docker 是否安装成功 3.2 创建 Docker 网络3.3 创建主从节点的 SQL S…

部署环境从docker swarm迁移到k8s后kie-server的发布方式变化(二)

正如后来的考虑, 如果外接maven私库照理说是ok的, 这样去掉volume的设计整个流程更加的自动化标准化. 开搞 第一步先部署一个nexus yaml文件如下: apiVersion: v1 kind: Namespace metadata:name: nexus---apiVersion: v1 kind: Service metadata:labels:k8s-app: nexusname…

【数据结构与算法】八大排序

[数据结构与算法]八大排序 数据结构与算法-八大排序排序的概念及其应用排序的概念排序的应用 常见的排序算法实现常见的排序算法插入排序直接插入排序希尔排序(缩小增量排序)希尔排序的时间复杂度希尔排序对插入排序的优化效果 选择排序直接选择排序堆排序向上调整建堆&#xf…

【C语言】从n个字符串中匹配查找abc字符串个数

目录 前言知识点重温1、scanf和scanf_s区别2、指针3、char、char*使用查找字符串收尾前言 这是小5聊的《C语言知识点例子》系列的第四篇文章。 在软件行业已经有快十年,技术虽然一般般,但是足够应付和解决编程入门的相关问题! 都说十年磨一剑,积累到一定经验,是时候发挥自…

数据库基础篇 《1. 概述》

目录 1. 为什么要使用数据库 2. 数据库与数据库管理系统 2.1 数据库的相关概念 2.2 数据库与数据库管理系统的关系 2.3 常见的数据库管理系统排名(DBMS) 2.4 常见的数据库介绍 3. MySQL介绍 3.1 概述 3.2 MySQL发展史重大事件 4. RDBMS 与 非RDBMS 4.1 关系型数据库…

Delphi Web Server 流程分析

通过向导 "Web Server Application" (选择 "Stand-alone GUI Application") 创建一个 WebServer Demo。 主单元代码: ...... private FServer: TIdHTTPWebBrokerBridge; procedure StartServer; ............. Delphi的网络组件是基于INDY的&a…

项目四:无极调光台灯

项目四&#xff1a;无极调光台灯 文章目录 项目四&#xff1a;无极调光台灯一、导入(5分钟&#xff09;学习目的 二、新授(65分钟)1.预展示结果(5分钟)2.本节课所用的软硬件(5分钟)3.硬件介绍(5分钟)4.图形化块介绍(10分钟)5.单个模块的简单使用(10分钟)6.无极调光台灯编程逻辑…

SQLServer的内存管理架构

内存管理架构说明 一、Windows的虚拟内存管理器二、SQL Server 内存体系结构2.1、传统&#xff08;虚拟&#xff09;内存2.2、地址窗口扩展 &#xff08;AWE&#xff09; 内存 三、从 SQL Server 2012 &#xff08;11.x&#xff09; 开始发生的改变3.1、对内存管理的更改3.2、对…

【网络原理】TCP/IP协议

目录 1.应用层 2.传输层&#xff08;核心问题&#xff09; 2.1 UDP协议 2.1.2 UDP的特点 2.1.3 基于UDP的应用层协议 2.2 TCP协议&#xff08;重点内容&#xff09; 2.2.1 TCP/IP 协议含义 2.2.2 TCP协议端格式&#xff1a; 2.2.3 TCP的特点 2.3 TCP原理 2.4 确认应…

Fork/Join优化mybatis-plus批量插入性能

最近在项目开发中&#xff0c;遇到需要一次性保存100万数据到数据库中。想到以下几种实现方式&#xff1a; 第一种方案&#xff1a;在mapper文件中&#xff0c;实现批量插入动态SQL语句&#xff0c;采用insert into table_name values(?,?,?,),(?,?,?)拼接SQL方式。 &l…

HTTP中ETag语法及使用实战详解

1.1 ETag 是什么 ETag&#xff08;Entity Tag&#xff09;是万维网协议 HTTP 的一部分。它是 HTTP 协议提供的若干机制中的一种 Web 缓存验证机制&#xff0c;并且允许客户端进行缓存协商。这使得缓存变得更加高效&#xff0c;而且节省带宽。如果资源的内容没有发生改变&#x…

电脑网速慢怎么解决?4个方法有效提升电脑网速!

案例&#xff1a;电脑网速慢怎么解决 【谁懂啊&#xff01;我的电脑网速太慢了&#xff01;总是上不了网&#xff0c;打开浏览器也是一直在转圈圈&#xff01;太折磨了&#xff01;这是为什么呢&#xff1f;谁能帮帮我呀&#xff01;】 随着互联网的发展和普及&#xff0c;电…

继续学c++

由于c里面有很多和c语言很像的东西&#xff0c;这里就来总结一点不像的或者要注意的&#xff0c;或者是我已经快忘记的&#xff1b; 先来一个浮点型也就是实型类型的总结&#xff1b; 知道浮点型有这两个类型&#xff1a;float和double型&#xff1b; 然后float型占四个字节…

To B第六年,腾讯过分温柔

腾讯做2B&#xff0c;方向是正确的&#xff0c;初心是果决的&#xff0c;行动是温柔的&#xff0c;事实是掉队的。 2018年&#xff0c;率先打出“互联网的下半场属于产业互联网”的大旗&#xff0c;宣布“拥抱产业互联网”&#xff0c;腾讯发力To B业务&#xff0c;绝对是有先发…

HTB-Jarvis

HTB-Jarvis 信息收集80端口 www-data(sqlmap)www-data(myPhpAdmin)www-data -> pepperpepper -> root 信息收集 80端口 目录扫描 我啥也没干咋就被ban了&#xff0c;可能是gobuster流量太大被逮住了。 老老实实等90秒&#xff0c;先从已有的目录收集信息。 phpMyAdmin 4…