【Android】Recyclerview的缓存复用
介绍
RecyclerView是Android开发中常用的一个高度可定制的列表视图组件。它是在ListView和GridView的基础上进行了改进和增强,旨在提供更好的性能和更灵活的布局管理。
RecyclerView的主要特点如下:
灵活的布局管理器(LayoutManager):
RecyclerView使用LayoutManager来管理其子视图的布局方式。不同的LayoutManager可以实现不同的布局效果,如线性布局、网格布局、瀑布流布局等。可以根据需要选择适合的LayoutManager或自定义LayoutManager。
高效的回收再利用机制:
RecyclerView通过Recycler来处理子视图的回收和再利用。当子视图滑出屏幕时,RecyclerView会将其回收并标记为可重用。这样可以减少内存占用,提高性能。
分离的ViewHolder模式:
RecyclerView使用ViewHolder模式来管理子视图的数据绑定。通过创建ViewHolder来保存子视图的引用,可以避免频繁的findViewById操作,提高列表项的渲染效率。
动画支持:
RecyclerView提供了对列表项的动画支持。可以通过设置ItemAnimator来实现添加、删除、移动等操作时的动画效果,为列表增添交互和视觉效果。
事件拦截与处理:
RecyclerView支持事件的拦截和处理,可以监听和处理子视图的点击、长按、滑动等事件,并根据需要进行相应的操作。
RecyclerView嵌套滑动
1.RecyclerView的嵌套滑动开始时,会调用dispatchNestedPreScroll方法,通知父容器即将发生的滑动事件。这是为了让父容器(例如外部的ScrollView或NestedScrollView)有机会对滑动事件进行预处理。
2.在dispatchNestedPreScroll之后,RecyclerView会调用scrollByInternal方法进行滚动操作。scrollByInternal方法会计算滚动的偏移量,并将其应用到RecyclerView的内容上。
3.接着,RecyclerView会调用dispatchNestedScroll方法,通知父容器滑动的实际情况,包括滑动的距离和方向等信息。父容器可以根据这些信息来进行相应的处理,例如触发边缘效果。
4.在dispatchNestedScroll方法之后,RecyclerView会调用scrollStep方法,其中包含滚动的水平和垂直偏移量。scrollStep方法会调用LayoutManager的scrollHorizontallyBy和scrollVerticallyBy方法,以实现具体的滚动效果。
5.scrollHorizontallyBy和scrollVerticallyBy是LayoutManager的内部方法,负责处理具体的水平和垂直滚动。它们会根据偏移量更新RecyclerView中可见视图的位置。
6.在滚动过程中,LayoutManager会调用layoutChunk方法来填充新的可见视图。在这个过程中,RecyclerView会调用recycler.getViewForPosition方法来获取指定位置的视图,并使用measureChildWithMargins方法来测量视图的尺寸。
7.measureChildWithMargins方法是用来测量RecyclerView中每个可见视图的尺寸的。它考虑到视图的padding、margin和inset(分割线的空间)等因素,确保视图能够正确地布局在RecyclerView中。
Fill方法
RecyclerView中的fill方法是用来填充可见视图的核心方法。在RecyclerView进行滚动时,fill方法会被反复调用,用于根据滚动方向和滚动偏移量来填充新的可见视图。
fill方法的主要逻辑:
1.首先,fill方法会根据滚动方向(垂直或水平)获取RecyclerView的布局状态(LayoutState),布局状态中包含了一些关键信息,比如当前可见的位置、偏移量等。
2.然后,fill方法会根据布局状态中的信息,调用LayoutManager的next方法获取下一个要填充的子视图。LayoutManager会根据当前的布局状态和回收池(Recycler)中的视图,选择合适的子视图并返回给fill方法。
3.接下来,fill方法会调用LayoutManager的measureChildWithMargins方法对子视图进行测量。这个方法会考虑到子视图的padding、margin和inset等因素,确保子视图能够正确地布局在RecyclerView中。
4.然后,fill方法会调用LayoutManager的layoutDecorated方法对子视图进行布局。layoutDecorated方法会设置子视图的位置和尺寸,将其正确地放置在RecyclerView中的指定位置。
5.填充完一个子视图后,fill方法会更新布局状态中的一些信息,比如当前可见的位置、偏移量等。
6.接着,fill方法会判断是否填充完所有可见位置的子视图。如果还有未填充的位置,fill方法会继续进行下一轮的填充。
通过不断调用fill方法,RecyclerView能够动态地填充滚动过程中新出现的可见视图,并回收滚出屏幕的不可见视图,实现高效的滚动和视图复用。
需要注意的是,fill方法的具体实现是由LayoutManager来定义的,不同的LayoutManager可能会有不同的实现逻辑。在自定义LayoutManager时,可以重写fill方法来实现特定的布局需求。
RecyclerView的多级缓存
mChangeScrap与 mAttachedScrap
mChangedScrap:该列表用于缓存仍然在屏幕内但【数据发生变化】的ViewHolder。当RecyclerView执行布局过程时,它会标记数据有变化的ViewHolder,并将它们添加到mChangedScrap中,以便在稍后重新绑定数据。
mAttachedScrap:这个列表用于缓存仍然在屏幕内且可见的ViewHolder。当RecyclerView滚动时,屏幕上新进入的ViewHolder会添加到mAttachedScrap中,以便在稍后进行布局和数据绑定。
ViewHoldermCachedViews
这个列表用于缓存已经离开屏幕的ViewHolder。当RecyclerView滚动时,屏幕上移出的ViewHolder会添加到mCachedViews中。这些ViewHolder被缓存起来,以便在需要时可以快速复用,而不必重新创建。
ViewHoldermViewCacheExtension
这个机制允许开发者自定义视图的缓存和复用逻辑。通过实现ViewCacheExtension接口,开发者可以指定一种特定类型的视图缓存方式,以提高复用效率。
RecycledViewPool
这是一个全局的ViewHolder缓存池。当RecyclerView无法从其他缓存列表中找到可复用的ViewHolder时,它会尝试从RecycledViewPool中获取。这个缓存池可以跨不同类型的ViewHolder进行复用,以提高整体的复用效率。
RecyclerView进行ViewHolder的回收和复用时,使用了以下几个关键的数据结构:
mCachedViews:
这是一个ArrayList,用于缓存已经离开屏幕的ViewHolder。RecyclerView默认情况下会为该列表分配2个ViewHolder的空间,即DEFAULT_CACHE_SIZE = 2。如果列表已满,最旧的ViewHolder将被移除。也可以使用setViewCacheSize方法来设置这个缓存列表的大小。
scrapHeap:
这也是一个ArrayList,用于存放回收池中的ViewHolder。回收池是全局的,供整个RecyclerView使用。RecyclerView默认情况下会为该列表分配5个ViewHolder的空间,即DEFAULT_MAX_SCRAP = 5。当需要复用ViewHolder时,首先尝试从回收池中获取ViewHolder。可以使用setMaxRecycledViews方法来设置回收池中ViewHolder的最大数量。
mScrap:
这是一个SparseArray,用于存储不同类型ViewHolder的回收数据。它是一个以整数作为键(viewType)的映射表,每个键对应一个ScrapData对象。ScrapData对象包含了该类型ViewHolder的回收池(mScrapHeap)。通过这种方式,RecyclerView能够针对不同的ViewHolder类型进行缓存和复用。
缓存复用 针对什么? 保存的是什么?
对于ViewHolder的缓存复用,针对的是不同类型的ViewHolder。保存的是已经创建过的、暂时不再需要显示在屏幕上的ViewHolder。
具体来说:
缓存池中复用ViewHolder(从回收池中获取):
当需要创建一个新的ViewHolder时,首先会尝试从回收池中获取对应类型的ViewHolder。回收池是一个用于缓存已创建的ViewHolder的池子。如果回收池中有可用的ViewHolder,就可以直接复用它,而无需重新创建。这个过程会调用onBindViewHolder方法,用于更新ViewHolder的数据和界面。
从缓存视图列表中复用ViewHolder:
如果回收池中没有可用的ViewHolder,那么就会尝试从缓存视图列表中获取。缓存视图列表是一个专门用于缓存离开屏幕的ViewHolder的列表,存储了一定数量的ViewHolder。如果成功从该列表中获取到可复用的ViewHolder,就可以直接使用,无需重新创建,并且也无需调用onBindViewHolder方法,因为这个ViewHolder之前已经显示过并绑定了数据。
如果无法从上述两个地方获取可复用的ViewHolder,才会调用onCreateViewHolder方法,创建一个新的ViewHolder实例,并通过onBindViewHolder方法绑定数据和界面。
综上所述,ViewHolder的缓存复用机制通过回收池和缓存视图列表来保存已经创建过的、暂时不再需要显示的ViewHolder实例。这样可以避免频繁地创建和销毁ViewHolder,提高RecyclerView的性能和效率。
相关文章:
【Android】Recyclerview的缓存复用
介绍 RecyclerView是Android开发中常用的一个高度可定制的列表视图组件。它是在ListView和GridView的基础上进行了改进和增强,旨在提供更好的性能和更灵活的布局管理。 RecyclerView的主要特点如下: 灵活的布局管理器(LayoutManager&#…...
机器学习:混合高斯聚类GMM(求聚类标签)+PCA降维(3维降2维)习题
使用混合高斯模型 GMM,计算如下数据点的聚类过程: Datanp.array([1,2,6,7]) 均值初值为: μ1,μ21,5 权重初值为: w1,w20.5,0.5 方差: std1,std21,1 K2 10 次迭代后数据的聚类标签是多少? 采用python代码实现: from scipy import…...
libuv库学习笔记-processes
Processes libuv提供了相当多的子进程管理函数,并且是跨平台的,还允许使用stream,或者说pipe完成进程间通信。 在UNIX中有一个共识,就是进程只做一件事,并把它做好。因此,进程通常通过创建子进程来完成不…...
c++ 给无名形参提供默认值
如上图,若函数的形参不在函数体里使用,可以不提供形参名,而且可以给此形参提供默认值。也能编译通过。 在看vs2019上的源码时,也出现了这种写法。应用SFINAE(substitute false is not an error)原则&#x…...
NO1.使用命令行创建Maven工程
①在工作空间目录下打开命令窗口 ②使用命令行生成Maven工程 mvn archetype:generate 运行 MVN 原型:生成命令,下面根据提示操作 选择一个数字或应用过滤器(格式:[groupId:]artifactId,区分大小写包含)&a…...
深度学习入门(一):神经网络基础
一、深度学习概念 1、定义 通过训练多层网络结构对位置数据进行分类或回归,深度学习解决特征工程问题。 2、深度学习应用 图像处理语言识别自然语言处理 在移动端不太好,计算量太大了,速度可能会慢 eg.医学应用、自动上色 3、例子 使用…...
网络知识整理
网络知识整理 网络拓扑网关默认网关 数据传输拓扑结构层面协议层面 网络拓扑 网关 连接两个不同的网络的设备都可以叫网关设备,网关的作用就是实现两个网络之间进行通讯与控制。 网关设备可以是交换机(三层及以上才能跨网络) 、路由器、启用了路由协议的服务器、代…...
如何有效地使用ChatGPT写小说讲故事?
构思故事情节,虽有趣但耗时,容易陷入写作瓶颈。ChatGPT可提供灵感,帮你解决写作难题。要写出引人入胜的故事,关键在于抓住八个要素——主题、人物、视角、背景、情节、语气、冲突和解决办法。 直接给出故事模板,你可…...
原生求生记:揭秘UniApp的原生能力限制
文章目录 1. 样式适配问题2. 性能问题3. 原生能力限制4. 插件兼容性问题5. 第三方组件库兼容性问题6. 全局变量污染7. 调试和定位问题8. 版本兼容性问题9. 前端生态限制10. 文档和支持附录:「简历必备」前后端实战项目(推荐:⭐️⭐️⭐️⭐️…...
网络编程 IO多路复用 [epoll版] (TCP网络聊天室)
//head.h 头文件 //TcpGrpSer.c 服务器端 //TcpGrpUsr.c 客户端 通过IO多路复用实现服务器在单进程单线程下可以与多个客户端交互 API epoll函数 #include<sys/epoll.h> int epoll_create(int size); 功能:创建一个epoll句柄//创建红黑树根…...
【go-zero】浅析 01
“github.com/google/uuid” uuid.New().String() go-zero 文档 https://www.w3cschool.cn/gozero/ go-zero 官网 https://go-zero.dev/ 快速开始: $ mkdir go-zero-demo $ cd go-zero-demo $ go mod init go-zero-demo $ goctl api new greet $ go mod tidy Done…...
音视频——视频流H264编码格式
1 H264介绍 我们了解了什么是宏快,宏快作为压缩视频的最小的一部分,需要被组织,然后在网络之间做相互传输。 H264更深层次 —》宏块 太浅了 如果单纯的用宏快来发送数据是杂乱无章的,就好像在没有集装箱 出现之前,…...
【使用深度学习的城市声音分类】使用从提取音频特征(频谱图)中提取的深度学习进行声音分类研究(Matlab代码实现)
💥💥💞💞欢迎来到本博客❤️❤️💥💥 🏆博主优势:🌞🌞🌞博客内容尽量做到思维缜密,逻辑清晰,为了方便读者。 ⛳️座右铭&a…...
机器学习完整路径
一个机器学习项目从开始到结束大致分为 5 步,分别是定义问题、收集数据和预处理、选择算法和确定模型、训练拟合模型、评估并优化模型性能。是一个循环迭代的过程,优秀的模型都是一次次迭代的产物。 定义问题 要剖析业务场景,设定清晰的目标…...
CK-00靶机详解
CK-00靶机详解 靶场下载地址:https://download.vulnhub.com/ck/CK-00.zip 这个靶场扫描到ip打开后发现主页面css是有问题的,一般这种情况就是没有配置域名解析。 我们网站主页右击查看源代码,发现一个域名。 把域名添加到我们hosts文件中。…...
17-C++ 数据结构 - 栈
📖 1.1 什么是栈 栈是一种线性数据结构,具有后进先出(Last-In-First-Out,LIFO)的特点。可以类比为装满盘子的餐桌,每次放盘子都放在最上面,取盘子时也从最上面取,因此最后放进去的盘…...
Redis如何实现排行榜?
今天给大家简单聊聊 Redis Sorted Set 数据类型底层的实现原理和游戏排行榜实战。特别简单,一点也不深入,也就 7 张图,粉丝可放心食用,哈哈哈哈哈~~~~。 1. 是什么 Sorted Sets 与 Sets 类似,是一种集合类型ÿ…...
Pycharm debug程序,跳转至指定循环条件/循环次数
在断点出右键,然后设置条件 示例 for i in range(1,100):a i 1b i 2print(a, b, i) 注意: 1、你应该debug断点在循环后的位置而不是循环上的位置,然后你就可以设置你的条件进入到指定的循环上了 2、设置条件,要使用等于符号…...
react实现markdown
参考:https://blog.csdn.net/Jack_lzx/article/details/118495763 参考:https://blog.csdn.net/m0_48474585/article/details/119742984 0. 示例 用react实现markdown编辑器 1.基本布局及样式 <><div classNametf_editor_header>头部&…...
HTTP请求走私漏洞简单分析
文章目录 HTTP请求走私漏洞的产生HTTP请求走私漏洞的分类HTTP请求走私攻击的危害确认HTTP请求走私漏洞通过时间延迟技术确认CL漏洞通过时间延迟技术寻找TE.CL漏洞 使用差异响应内容确认漏洞通过差异响应确认CL.TE漏洞通过差异响应确认TE.CL漏洞 请求走私漏洞的利用通过请求漏洞…...
生成xcframework
打包 XCFramework 的方法 XCFramework 是苹果推出的一种多平台二进制分发格式,可以包含多个架构和平台的代码。打包 XCFramework 通常用于分发库或框架。 使用 Xcode 命令行工具打包 通过 xcodebuild 命令可以打包 XCFramework。确保项目已经配置好需要支持的平台…...
ES6从入门到精通:前言
ES6简介 ES6(ECMAScript 2015)是JavaScript语言的重大更新,引入了许多新特性,包括语法糖、新数据类型、模块化支持等,显著提升了开发效率和代码可维护性。 核心知识点概览 变量声明 let 和 const 取代 var…...
智慧工地云平台源码,基于微服务架构+Java+Spring Cloud +UniApp +MySql
智慧工地管理云平台系统,智慧工地全套源码,java版智慧工地源码,支持PC端、大屏端、移动端。 智慧工地聚焦建筑行业的市场需求,提供“平台网络终端”的整体解决方案,提供劳务管理、视频管理、智能监测、绿色施工、安全管…...
MFC内存泄露
1、泄露代码示例 void X::SetApplicationBtn() {CMFCRibbonApplicationButton* pBtn GetApplicationButton();// 获取 Ribbon Bar 指针// 创建自定义按钮CCustomRibbonAppButton* pCustomButton new CCustomRibbonAppButton();pCustomButton->SetImage(IDB_BITMAP_Jdp26)…...
Cilium动手实验室: 精通之旅---20.Isovalent Enterprise for Cilium: Zero Trust Visibility
Cilium动手实验室: 精通之旅---20.Isovalent Enterprise for Cilium: Zero Trust Visibility 1. 实验室环境1.1 实验室环境1.2 小测试 2. The Endor System2.1 部署应用2.2 检查现有策略 3. Cilium 策略实体3.1 创建 allow-all 网络策略3.2 在 Hubble CLI 中验证网络策略源3.3 …...
Python实现prophet 理论及参数优化
文章目录 Prophet理论及模型参数介绍Python代码完整实现prophet 添加外部数据进行模型优化 之前初步学习prophet的时候,写过一篇简单实现,后期随着对该模型的深入研究,本次记录涉及到prophet 的公式以及参数调优,从公式可以更直观…...
Qt Http Server模块功能及架构
Qt Http Server 是 Qt 6.0 中引入的一个新模块,它提供了一个轻量级的 HTTP 服务器实现,主要用于构建基于 HTTP 的应用程序和服务。 功能介绍: 主要功能 HTTP服务器功能: 支持 HTTP/1.1 协议 简单的请求/响应处理模型 支持 GET…...
零基础设计模式——行为型模式 - 责任链模式
第四部分:行为型模式 - 责任链模式 (Chain of Responsibility Pattern) 欢迎来到行为型模式的学习!行为型模式关注对象之间的职责分配、算法封装和对象间的交互。我们将学习的第一个行为型模式是责任链模式。 核心思想:使多个对象都有机会处…...
深入解析C++中的extern关键字:跨文件共享变量与函数的终极指南
🚀 C extern 关键字深度解析:跨文件编程的终极指南 📅 更新时间:2025年6月5日 🏷️ 标签:C | extern关键字 | 多文件编程 | 链接与声明 | 现代C 文章目录 前言🔥一、extern 是什么?&…...
Android 之 kotlin 语言学习笔记三(Kotlin-Java 互操作)
参考官方文档:https://developer.android.google.cn/kotlin/interop?hlzh-cn 一、Java(供 Kotlin 使用) 1、不得使用硬关键字 不要使用 Kotlin 的任何硬关键字作为方法的名称 或字段。允许使用 Kotlin 的软关键字、修饰符关键字和特殊标识…...
