深入源码分析RecyclerView缓存复用原理

news2024/11/24 6:18:50

文章目录

    • 前言
      • 四级缓存
    • 源码分析
      • 缓存
        • 一级缓存(mChangedScrap和mChangedScrap)
        • 二级缓存(mCachedViews)
        • 三级缓存
        • 四级缓存(mRecyclerPool)
          • 缓存池mRecyclerPool结构理解
          • 四级缓存简单小结
        • 缓存流程图
      • 复用
        • 复用流程图
    • 结语

前言

RecyclerView是Android日常开发中经常使用的控件,了解其源码,明白其中的缓存复用机制是十分有必要的;

四级缓存

我们都知道RecyclerView有四级缓存,缓存的都是ViewHolder对象,那都分别对应哪些缓存呢?各自缓存的作用是什么呢?这里先简单总结下:

层级缓存变量容量数据结构作用
1mChangedScrap与 mAttachedScrapXArrayList<ViewHolder>用来缓存还在屏幕内的ViewHolder
2mCachedViews默认为2,可通过调用setViewCacheSize()方法调整ArrayList<ViewHolder>用来缓存移除屏幕之外的ViewHolder
3mViewCacheExtensionX自定义缓存,一般不使用
4mRecyclerPool每个itemViewType默认存储5个ViewHolderSparseArray<ScrapData>ViewHolder缓存池,复用时需要重新调用onBindViewHolder

其中ScrapData结构如下:

      static class ScrapData {
            final ArrayList<ViewHolder> mScrapHeap = new ArrayList<>();
        }

源码分析

缓存

我们从RecyclerViewonLayout方法开始跟踪:

    protected void onLayout(boolean changed, int l, int t, int r, int b) {
    	...
        dispatchLayout();
   		...
    }

其中dispatchLayout()方法如下:

 void dispatchLayout() {
        if (mAdapter == null) {
            Log.e(TAG, "No adapter attached; skipping layout");
            return;
        }
        if (mLayout == null) {
            Log.e(TAG, "No layout manager attached; skipping layout");
            return;
        }
        mState.mIsMeasuring = false;
        if (mState.mLayoutStep == State.STEP_START) {
        	//dispatchLayoutStep1()中会做以下几件事:
        	1.处理适配器的更新;
        	2.决定应该运行哪个动画;
        	3.保存有关当前视图的信息;
        	4.运行预测布局并保存其信息;
            dispatchLayoutStep1();
            mLayout.setExactMeasureSpecsFrom(this);
            //dispatchLayoutStep2()中会进行实际的布局操作
            dispatchLayoutStep2();
        } else if (mAdapterHelper.hasUpdates() || mLayout.getWidth() != getWidth()
                || mLayout.getHeight() != getHeight()) {
            // 当宽高改变时,会再次调用 dispatchLayoutStep2()进行重新布局;
            mLayout.setExactMeasureSpecsFrom(this);
            dispatchLayoutStep2();
        } else {
            // always make sure we sync them (to ensure mode is exact)
            mLayout.setExactMeasureSpecsFrom(this);
        }
        //dispatchLayoutStep3()处理相关动画
        dispatchLayoutStep3();
    }

这里我们重点关注下 dispatchLayoutStep2()方法;

    private void dispatchLayoutStep2() {
     	...
        mLayout.onLayoutChildren(mRecycler, mState);
        ...
    }

显然,由于dispatchLayoutStep2()主要工作是重新布局,那么肯定要进行子View的布局;
其中 mLayout.onLayoutChildren(mRecycler, mState);调用的是LayoutManager的onLayoutChildren方法,
这里,我们选择LinearLayoutManager来跟进流程;

### LinearLayoutManager.onLayoutChildren
    public void onLayoutChildren(RecyclerView.Recycler recycler, RecyclerView.State state) {
        ...
        detachAndScrapAttachedViews(recycler);
       	...
    }

onLayoutChildren会调用detachAndScrapAttachedViews(recycler)方法

     public void detachAndScrapAttachedViews(@NonNull Recycler recycler) {
            final int childCount = getChildCount();
            for (int i = childCount - 1; i >= 0; i--) { 
                final View v = getChildAt(i);
                scrapOrRecycleView(recycler, i, v);
            }
        }

注意这里是倒序遍历,我们重点看看scrapOrRecycleView(recycler, i, v);方法;

   final ViewHolder viewHolder = getChildViewHolderInt(view);
   			//如果viewHolder设置成ignore,则直接返回;
            if (viewHolder.shouldIgnore()) { 
                if (DEBUG) {
                    Log.d(TAG, "ignoring view " + viewHolder);
                }
                return;
            }
            //如果viewHolder数据非法无效 && viewHolder不指向数据集中移除的数据 && adapter没有设置stableId 
            if (viewHolder.isInvalid() && !viewHolder.isRemoved()
                    && !mRecyclerView.mAdapter.hasStableIds()) {
                 //移除当前子View
                removeViewAt(index);
                //里面会调用mCachedViews和mRecyclerPool进行二级和四级缓存(三级缓存为自定义缓存)
                recycler.recycleViewHolderInternal(viewHolder);
            } else {
            	//暂时将View解绑,以便后续可以通过ViewHolder重新绑定复用
                detachViewAt(index);
                //里面会根据条件调用mAttachedScrap或mChangedScrap进行一级缓存;
                recycler.scrapView(view);
                //从消失列表中移除viewHolder
                mRecyclerView.mViewInfoStore.onViewDetached(viewHolder);
            }

接下来,我们就重点分别看recycler.scrapView(view) recycler.recycleViewHolderInternal(viewHolder)方法;

一级缓存(mChangedScrap和mChangedScrap)

    void scrapView(View view) {
            final ViewHolder holder = getChildViewHolderInt(view);
                //如果ViewHolder标记为移除或失效的 || ViewHolder没有变化 || item 无动画或动画不复用
            if (holder.hasAnyOfTheFlags(ViewHolder.FLAG_REMOVED | ViewHolder.FLAG_INVALID) || !holder.isUpdated() || canReuseUpdatedViewHolder(holder)) {
        
               if (holder.isInvalid() && !holder.isRemoved() && !mAdapter.hasStableIds()) {
                    throw new IllegalArgumentException("Called scrap view with an invalid view."));
                }
                holder.setScrapContainer(this, false);
                mAttachedScrap.add(holder);
            } else {
                if (mChangedScrap == null) {
                    mChangedScrap = new ArrayList<ViewHolder>();
                }
                holder.setScrapContainer(this, true);
                mChangedScrap.add(holder);
            }
        }

从上述代码可以看出:当ViewHolder满足移除或失效||没有变化||没有动画或动画不复用时,缓存到mAttachedScrap集合中,否则缓存到mChangedScrap集合中;

二级缓存(mCachedViews)

        void recycleViewHolderInternal(ViewHolder holder) {
           	...
            if (forceRecycle || holder.isRecyclable()) {
                if (mViewCacheMax > 0
                        && !holder.hasAnyOfTheFlags(ViewHolder.FLAG_INVALID
                        | ViewHolder.FLAG_REMOVED
                        | ViewHolder.FLAG_UPDATE
                        | ViewHolder.FLAG_ADAPTER_POSITION_UNKNOWN)) {
                    // 先获取mCachedViews的大小
                    int cachedViewSize = mCachedViews.size();
                    //如果mCachedViews大小超过或等于默认值2的时候
                    if (cachedViewSize >= mViewCacheMax && cachedViewSize > 0) {
                        recycleCachedViewAt(0);//将下标为0位置的元素从集合中移除,放入到四级缓存mRecyclerPool中
                        cachedViewSize--; //集合大小-1
                    }
					
                    int targetCacheIndex = cachedViewSize; //将cachedViewSize赋值给targetCacheIndex
                    if (ALLOW_THREAD_GAP_WORK
                            && cachedViewSize > 0
                            && !mPrefetchRegistry.lastPrefetchIncludedPosition(holder.mPosition)) {
    				...
    				//缓存新的holder至targetCacheIndex下标中,并设置cached为true
                    mCachedViews.add(targetCacheIndex, holder);
                    cached = true;
                }
                if (!cached) {
               		//没有缓存成功,则放入到四级缓存mRecyclerPool中
                    addViewHolderToRecycledViewPool(holder, true);
                    recycled = true;
                }
            } 
            ...
        }

从上述代码中可以看出:当满足移除屏幕条件时:
1. 当mCachedViews没满时,ViewHolder会直接缓存到mCachedViews中,如果缓存失败,则会缓存到四级缓存mRecyclerPool中;

2. 当mCachedViews满时,会先移除mCachedViews集合中下标为0位置的元素,并将其放置到缓存池mRecyclerPool中;然后将ViewHolder缓存到mCachedViews集合下标为1位置上,如果缓存失败,则会缓存到四级缓存mRecyclerPool中;

三级缓存

为用户自定义缓存,可通过自定义ViewCacheExtension,并重写getViewForPositionAndType方法实现;

四级缓存(mRecyclerPool)

从上面二级缓存实现可以看到,会调用addViewHolderToRecycledViewPool(holder, true)实现四级缓存机制;

  void addViewHolderToRecycledViewPool(@NonNull ViewHolder holder, boolean dispatchRecycled) {
  			//1.将viewHolder引用的recyclerView移除掉
            clearNestedRecyclerViewIfNotNested(holder);
           	...
           	//2.移除viewHolder相关监听
            if (dispatchRecycled) {
                dispatchViewRecycled(holder);
            }
            holder.mOwnerRecyclerView = null;
            //3.缓存至mRecyclerPool中;
            getRecycledViewPool().putRecycledView(holder);
        }
		###  getRecycledViewPool().putRecycledView
		
        public void putRecycledView(ViewHolder scrap) {
        	//1.先获取ViewHolder对象的itemViewType
            final int viewType = scrap.getItemViewType();
            //2.根据itemViewType获取对应的ArrayList<ViewHolder>集合
            final ArrayList<ViewHolder> scrapHeap = getScrapDataForType(viewType).mScrapHeap;
            //3.如果集合中已经保存有5个ViewHolder了,那就不再进行缓存操作;
            if (mScrap.get(viewType).mMaxScrap <= scrapHeap.size()) {
                return;
            }
            //4.已经缓存的有,抛异常
            if (DEBUG && scrapHeap.contains(scrap)) {
                throw new IllegalArgumentException("this scrap item already exists");
            }
            //5.将ViewHolder进行`漂白`,清除相关标志、位置信息等等,因此复用缓存池中的ViewHolder需要重新进行绑定操作;
            scrap.resetInternal();
            //6.添加到缓冲池中;
            scrapHeap.add(scrap);
        }
缓存池mRecyclerPool结构理解

缓存池mRecyclerPool结构理解

四级缓存简单小结

根据ViewHolder对应的itemViewType从缓存池中获取对应的ScrapData对象,ScrapData对象内部存储了ArrayList<ViewHolder> 集合,如果当前集合已满5个,则丢弃ViewHolder不进行缓存,如果集合不满,则先将ViewHolder进行数据漂白,清除相关信息后再添加到缓存集合中!

缓存流程图

缓存流程图

复用

复用流程图

结语

如果以上文章对您有一点点帮助,希望您不要吝啬的点个赞加个关注,您每一次小小的举动都是我坚持写作的不懈动力!ღ( ´・ᴗ・` )

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

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

相关文章

分布式项目15 用户注册,单点登陆dubbo来实现

分析&#xff1a;当用户填写完成注册信息之后,将请求发送给前台服务器.之后前台消费者利用dubbo框架实现RPC调用。之后将用户信息传递给jt-sso服务提供者.之后完成数据的入库操作。 01.页面url分析 02.查看页面JS $.ajax({ type : "POST", url : "/user/doRe…

MMPose安装及推理验证

MMPose安装 依赖环境 1.创建虚拟环境并激活 conda create --name openmmlab python3.8 -y conda activate openmmlab2.安装pytorch conda install pytorch torchvision torchaudio pytorch-cuda11.7 -c pytorch -c nvidia报错&#xff1a; InvalidArchiveError(‘Error wit…

quartus下联合modelsim_Altera仿真

vivado工程转换到quartus下联合modelsim仿真_内有小猪卖的博客-CSDN博客 这个博客是用单独的modelsim仿真&#xff0c;而下面的流程是使用quartus自带的modelsim-altera仿真。 版本为&#xff1a;quartus ii 13.1 64-bit 以fpga实现数码管和流水灯编码为例。数码管为1时&#x…

Spring中Bean加载流程

上面是跟踪了 getBean 的调用链创建的流程图&#xff0c;为了能够很好地理解 Bean 加载流程&#xff0c;省略一些异常、日志和分支处理和一些特殊条件的判断。 从上面的流程图中&#xff0c;可以看到一个 Bean 加载会经历这么几个阶段&#xff08;用绿色标记&#xff09;&…

机器学习算法分类(三)

在机器学习中&#xff0c;又分为监督学习、无监督学习、半监督学习、强化学习和深度学习。 监督、无监督、半监督学习 机器学习根据数据集是否有标签&#xff0c;又分为监督学习、无监督学习、半监督学习。 监督学习&#xff1a;训练数据集全部都有标签无监督学习&#xff1a…

grep命令的使用

grep命令是Linux中常用的文本搜索工具&#xff0c;它可以根据用户指定的模式&#xff0c;在文件或标准输入中查找匹配的文本行并返回。 下面是grep命令的一些常见选项&#xff1a; -i&#xff1a;忽略大小写-n&#xff1a;显示匹配行的行号-v&#xff1a;显示不匹配的行-r&am…

【新版】系统架构设计师 - 计算机系统基础知识

个人总结&#xff0c;仅供参考&#xff0c;欢迎加好友一起讨论 文章目录 架构 - 计算机系统基础知识考点摘要计算机系统计算机硬件组成浮点数Flynn分类法CISC与RISC流水线技术超标量流水线存储系统层次化存储结构CacheCache的命中率Cache的页面淘汰主存编址磁盘管理&#xff08…

《MySQL(四):基础篇- 约束》

文章目录 4. 约束4.1 概述4.2 约束演示4.3 外键约束4.3.1 介绍4.3.2 语法4.3.3 删除/更新行为 4. 约束 4.1 概述 概念&#xff1a;约束是作用于表中字段上的规则&#xff0c;用于限制存储在表中的数据。 目的&#xff1a;保证数据库中数据的正确、有效性和完整性。 分类: 约…

YOLOv5白皮书-第Y6周:模型改进

目录 一、改进网络结构设计1 改进的注意力机制2 多尺度特征融合3 改进的激活函数 二 数据增强和数据平衡1 数据增强2 数据平衡3 注意事项 三、模型融合策略1 投票策略2 加权平均策略3 特征融合策略4 其他模型融合策略 &#x1f368; 本文为&#x1f517;365天深度学习训练营 中…

protobuf实现原理

文章目录 一、前言二、概述三、数据存储方式&#xff1a;Varints(一)原理(二)举例(三)缺点 四、协议的数据结构(一)原理(二)举例 一、前言 最近刚刚从一家公司离职&#xff0c;在职的时候使用到了go语言的grpc库&#xff0c;了解了除了json之外的另一个专门用于远程调用的序列…

二本计算机专业学长经验之谈

2023.6.9 今年的行情对我们这些双非大学、二本真的太难了&#xff0c;菜鸟今年感觉毕业找的工作真的又苦逼钱又少&#xff0c;准备跳槽的&#xff0c;结果满大街投简历&#xff0c;连个毛都没有&#xff0c;唯一一个给了个海笔&#xff0c;然后就没然后… 所以希望大家真的要好…

Element的Select分组全选模式

Select 选择器选择器的分组&#xff0c;如上图所示&#xff0c;我们希望做到的效果是&#xff0c;点击“热门城市”或“城市名”的时候全选分组的options。 思路 思路一&#xff1a;目前的Select 选择器分组OptionGroup的Title只是一个文本DOM&#xff0c;没用其他东西&#…

详解基于罗德里格斯(Rodrigues)公式由旋转向量到旋转矩阵的 Python 实现

文章目录 旋转向量 rotation vector旋转矩阵 rotation matrix罗德里格斯公式 Rodrigues formula基于 Python 和 NumPy 实现 Rodrigues 公式 旋转向量 rotation vector 任何一个旋转都可以通过一个 旋转轴 加一个 旋转角 进行描述, 即围绕 旋转轴 旋转一个 旋转角. 此时可以通过…

javascript 中的 URL 解码

文章目录 需要URL编解码JavaScript 中的 URL 解码使用 unescaped() 方法解码编码的 URL使用 decodeURI() 方法解码编码的 URL使用 decodeURIComponent() 方法解码编码的 URL 总结 本文着眼于 URL 解码以及如何使用 JavaScript 对编码的 URL 进行解码。 需要URL编解码 URL 应具…

政企HTTPS加密国产化替代的四要素

信创产业是数字经济、信息安全发展的基础&#xff0c;也是“新基建”的重要内容&#xff0c;将成为拉动中国经济增长的重要抓手之一。随着国资委79号文的发布&#xff0c;国央企落实信息化系统的信创国产化改造的步伐加快&#xff0c;贯彻“28N”战略&#xff0c;从党政机关扩展…

Doris学习笔记

1.数据模型 数据模型 - Apache Doris 1.1 Aggregate 模型(聚合&#xff09; 可以发现&#xff0c;user_id、date、age ...等没有设置 AggregationType, 那么这几个字段就成了一个key了。设置了 AggregationType 字段&#xff0c;说明该列的属性已经成value了。 我们导入一张…

Linux·Binder机制原理

目录 前言 目录 1. Binder到底是什么&#xff1f; 2. 知识储备 2.1 进程空间划分 2.2 进程隔离 & 跨进程通信&#xff08; IPC &#xff09; 2.5 内存映射 3. Binder 跨进程通信机制 模型 3.1 模型原理图 3.3 模型原理步骤说明 3.4 额外说明 4. Binder机制 在An…

自学黑客(网络安全),一般人我劝你还是算了

写在开篇 笔者本人 17 年就读于一所普通的本科学校&#xff0c;20 年 6 月在三年经验的时候顺利通过校招实习面试进入大厂&#xff0c;现就职于某大厂安全联合实验室。 我为啥说自学黑客&#xff0c;一般人我还是劝你算了吧&#xff01;因为我就是那个不一般的人。 ​ 首先我…

elementui tree 支持虚拟滚动和treeLine (下)

​ 由于我之前没有发布过npm 包&#xff0c;这里还得现学一下。 参考资料&#xff1a; 链接: 如何写一个vue组件发布到npm&#xff0c;包教包会&#xff0c;保姆级教学链接: vue组件发布npm最佳实践 按照上面的步骤&#xff0c;我通过 vue-sfc-rollup 生成了项目&#xff0c;…

六级备考8天|CET-6|阅读强化|16:00~17:20

调整做题顺序&#xff1a;仔细阅读——>长篇阅读(信息匹配)——>翻译——>选词填空 顺关系 or 反关系 正态度 or 负态度 阅读要有针对性 理解要有空白性 2&#xff09;高大上的思维来自于中文的语言特点 练习 第一段&#xff1a;例子&#xff0c;无观点&am…