【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漏洞 请求走私漏洞的利用通过请求漏洞…...
浏览器访问 AWS ECS 上部署的 Docker 容器(监听 80 端口)
✅ 一、ECS 服务配置 Dockerfile 确保监听 80 端口 EXPOSE 80 CMD ["nginx", "-g", "daemon off;"]或 EXPOSE 80 CMD ["python3", "-m", "http.server", "80"]任务定义(Task Definition&…...
Debian系统简介
目录 Debian系统介绍 Debian版本介绍 Debian软件源介绍 软件包管理工具dpkg dpkg核心指令详解 安装软件包 卸载软件包 查询软件包状态 验证软件包完整性 手动处理依赖关系 dpkg vs apt Debian系统介绍 Debian 和 Ubuntu 都是基于 Debian内核 的 Linux 发行版ÿ…...
uni-app学习笔记二十二---使用vite.config.js全局导入常用依赖
在前面的练习中,每个页面需要使用ref,onShow等生命周期钩子函数时都需要像下面这样导入 import {onMounted, ref} from "vue" 如果不想每个页面都导入,需要使用node.js命令npm安装unplugin-auto-import npm install unplugin-au…...
今日学习:Spring线程池|并发修改异常|链路丢失|登录续期|VIP过期策略|数值类缓存
文章目录 优雅版线程池ThreadPoolTaskExecutor和ThreadPoolTaskExecutor的装饰器并发修改异常并发修改异常简介实现机制设计原因及意义 使用线程池造成的链路丢失问题线程池导致的链路丢失问题发生原因 常见解决方法更好的解决方法设计精妙之处 登录续期登录续期常见实现方式特…...
初学 pytest 记录
安装 pip install pytest用例可以是函数也可以是类中的方法 def test_func():print()class TestAdd: # def __init__(self): 在 pytest 中不可以使用__init__方法 # self.cc 12345 pytest.mark.api def test_str(self):res add(1, 2)assert res 12def test_int(self):r…...
LeetCode - 199. 二叉树的右视图
题目 199. 二叉树的右视图 - 力扣(LeetCode) 思路 右视图是指从树的右侧看,对于每一层,只能看到该层最右边的节点。实现思路是: 使用深度优先搜索(DFS)按照"根-右-左"的顺序遍历树记录每个节点的深度对于…...
算法:模拟
1.替换所有的问号 1576. 替换所有的问号 - 力扣(LeetCode) 遍历字符串:通过外层循环逐一检查每个字符。遇到 ? 时处理: 内层循环遍历小写字母(a 到 z)。对每个字母检查是否满足: 与…...
[免费]微信小程序问卷调查系统(SpringBoot后端+Vue管理端)【论文+源码+SQL脚本】
大家好,我是java1234_小锋老师,看到一个不错的微信小程序问卷调查系统(SpringBoot后端Vue管理端)【论文源码SQL脚本】,分享下哈。 项目视频演示 【免费】微信小程序问卷调查系统(SpringBoot后端Vue管理端) Java毕业设计_哔哩哔哩_bilibili 项…...
并发编程 - go版
1.并发编程基础概念 进程和线程 A. 进程是程序在操作系统中的一次执行过程,系统进行资源分配和调度的一个独立单位。B. 线程是进程的一个执行实体,是CPU调度和分派的基本单位,它是比进程更小的能独立运行的基本单位。C.一个进程可以创建和撤销多个线程;同一个进程中…...
Qemu arm操作系统开发环境
使用qemu虚拟arm硬件比较合适。 步骤如下: 安装qemu apt install qemu-system安装aarch64-none-elf-gcc 需要手动下载,下载地址:https://developer.arm.com/-/media/Files/downloads/gnu/13.2.rel1/binrel/arm-gnu-toolchain-13.2.rel1-x…...
