当前位置: 首页 > news >正文

recyclerview嵌套recyclerview_阿里3轮面试都问了RecyclerView

阿里面试总共4轮,其中有3轮面试都问到了RecyclerView的问题。面试的点各不相同,有原理、嵌套问题、有缓存实现,但是最终都是殊途同归,所有的问题都汇集在,如何对RecyclerView做性能优化?

1.1 RecyclerView第一次layout时,会发生预布局pre-layout吗?

​ 第一次布局时,并不会触发pre-layout。pre-layout只会在每次notify change时才会被触发,目的是通过saveOldPosition方法将屏幕中各位置上的ViewHolder的坐标记录下来,并在重新布局之后,通过对比实现Item的动画效果。比如以下效果:

f109ea4e00f7048cd773f48b59cf733c.png

1.2 如果自定义LayoutManager需要注意什么?

在RecyclerView的dispatchLayoutStep1阶段,会调用自定义LayoutManager的 supportsPredictiveItemAnimations 方法判断在某些状态下是否展示predictive animation。以下LinearLayoutManager的实现:

@Overridepublic boolean supportsPredictiveItemAnimations() {return mPendingSavedState == null && mLastStackFromEnd == mStackFromEnd;}

​ 如果 supportsPredictiveItemAnimations 返回true,则LayoutManager中复写onLayoutChildren方法会被调用2次:一次是在pre-layout,另一次是real-layout。

因为会有pre-layout和real-layout,所以在自定义LayoutManager中,需要根据RecyclerView.State中的isPreLayout方法的返回值,在这两次布局中做区分。比如LinearLayoutManager中的onLayoutChildren中有如下判断:

9c16e29ae5e82ec145cb6df11205b736.png

上面代码中有一段注释:

if the child is visible and we are going to move it around, we should layout extra items in the opposite direction to make sure new items animate nicely instead of just fading in

​ 代表的意思就是如果当前正在update的item是可见状态,则需要在pre-layout阶段额外填充一个item,目的是为了保证处于不可见状态的item可以平滑的滑动到屏幕内。

1.3 举例说明

​ 比如下图中点击item2将其删除,调用notifyItemRemoved后,在pre-layout之前item5并没有被添加到RecyclerView中,而经过pre-layout之后,item5经过布局会被填充到RecyclerView中

3ff46183af8a8113f5ea18e51dbc2bd6.png

当item移出屏幕之后,item5会随同item3和item4一起向上移动,如下图所示:

ce7118ece44f6804241cea32da62d34a.gif

​ 如果自定义LayoutManager并没有实现pre-layout,或者实现不合理,则当item2移出屏幕时,只会将item3和item4进行平滑移动,而item5只是单纯的appear到屏幕中,如下所示:

b9db766b5c84ce0c531605db708a9602.gif

可以看出item5并没有同item3和item4一起平滑滚动到屏幕内,这样界面上显示会给用户卡顿的感觉。

1.4 ViewHolder何时被缓存到RecycledViewPool中?

主要有以下2种情况:

  1. 当ItemView被滑动出屏幕时,并且CachedView已满,则ViewHolder会被缓存到RecycledViewPool中
  2. 当数据发生变动时,执行完disappearrance的ViewHolder会被缓存到RecycledViewPool中

1.5 CachedView和RecycledViewPool的关系

​ 当一个ItemView被滑动滚出屏幕之后,默认会先被保存在CachedView中。CachedView的默认大小为2,可以通过 setItemViewCacheSize 方法修改它的值。当CachedView已满后,后续有新的ItemView从屏幕内滑出时,会迫使CachedView根据FIFO规则,将之前的缓存的ViewHolder转移到RecycledViewPool中,效果可以参考下图:

dc45c14f080120327b3c7031ad1e21e9.gif

RecycledViewPool默认大小为5,可以通过以下方式修改RecycledViewPool的缓存大小:

RecyclerView.getRecycledViewPool().setMaxRecycledViews(int viewType, int max);

1.6 CachedView和RecycledViewPool两者区别

缓存到CachedView中的ViewHolder并不会清理相关信息(比如position、state等),因此刚移出屏幕的ViewHolder,再次被移回屏幕时,只要从CachedView中查找并显示即可,不需要重新绑定(bindViewHolder)。

而缓存到RecycledViewPool中的ViewHolder会被清理状态和位置信息,因此从RecycledViewPool查找到ViewHolder,需要重新调用bindViewHolder绑定数据。

1.7 你是从哪些方面优化RecyclerView的?

我总结了几点,主要可以从以下几个方面对RecyclerView进行优化:

尽量将复杂的数据处理操作放到异步中完成。RecyclerView需要展示的数据经常是从远端服务器上请求获取,但是在网络请求拿到数据之后,需要将数据做扁平化操作,尽量将最优质的数据格式返回给UI线程。

优化RecyclerView的布局,避免将其与ConstraintLayout使用

针对快速滑动事件,可以使用addOnScrollListener添加对快速滑动的监听,当用户快速滑动时,停止加载数据操作。

如果ItemView的高度固定,可以使用setHasFixSize(true)。这样RecyclerView在onMeasure阶段可以直接计算出高度,不需要多次计算子ItemView的高度,这种情况对于垂直RecyclerView中嵌套横向RecyclerView效果非常显著。

当UI是Tab feed流时,可以考虑使用RecycledViewPool来实现多个RecyclerView的缓存共享。

另外,我分享一份从网络上收录整理的 Android学习PDF+架构视频+面试文档等 ,还有Android开发面试专题资料,高级进阶架构资料供大家学习进阶。

02836fea8bdf1fbab9a6247b99025112.png
38ccd439083946b2a0100fc1caaf3be8.png
d10156898a424b9c8ce8e5977070b362.png

如果你需要《Android开发核心知识点笔记》最新版和Android学习PDF+架构视频+面试文档资料的话。帮忙转发+点赞,然后私信我【666】获取

喜欢本文的话,不妨给我点个小赞、评论区留言或者转发支持一下呗~

49bf13abd6586d522693a7b0771acd85.png

《Android开发核心知识点笔记》

相关文章:

  • 更改某用户个人网站的配额设置
  • python语音录音_同时从麦克风录制音频并用python播放
  • 隐藏自定义列表中的某些栏
  • python生成多级文件夹_利用 python 遍历多级文件夹处理不同文件
  • python去除视频中的logo_视频去水印神器(Easy Video Logo Remover)
  • MOSS 2007基础:内容类型(Content Type)
  • python 第三方包自动导入_Python第三方包的导入
  • SharePoint高级内容--访问群体对象模型的开发之一
  • php soap 两个版本_zabbix-第一章-第三节 LNMP之PHP搭建
  • SharePoint高级内容--访问群体对象模型的开发之二
  • 批量修改的后台代码_如何用WordPress自带的批量编辑功能修改文章分类目录?
  • postman压力测试_如何用Postman做接口自动化测试?
  • 编程访问“我的网站”中的目标链接门户列表
  • 你觉得外观模式和代理模式的联系和区别是什么?_清华毕业大牛带你深入研磨并掌握23种设计模式,总计6.17G...
  • MOSS 2007基础:开发自定义WebPart
  • 《剑指offer》分解让复杂问题更简单
  • 【笔记】你不知道的JS读书笔记——Promise
  • 11111111
  • Cookie 在前端中的实践
  • es的写入过程
  • Javascripit类型转换比较那点事儿,双等号(==)
  • k个最大的数及变种小结
  • scala基础语法(二)
  • use Google search engine
  • 基于Android乐音识别(2)
  • 码农张的Bug人生 - 初来乍到
  • 浅谈Kotlin实战篇之自定义View图片圆角简单应用(一)
  • 说说动画卡顿的解决方案
  • 以太坊客户端Geth命令参数详解
  • ###C语言程序设计-----C语言学习(3)#
  • $.ajax,axios,fetch三种ajax请求的区别
  • (42)STM32——LCD显示屏实验笔记
  • (arch)linux 转换文件编码格式
  • (顶刊)一个基于分类代理模型的超多目标优化算法
  • (附源码)springboot高校宿舍交电费系统 毕业设计031552
  • (附源码)ssm基于微信小程序的疫苗管理系统 毕业设计 092354
  • (没学懂,待填坑)【动态规划】数位动态规划
  • (免费领源码)Python#MySQL图书馆管理系统071718-计算机毕业设计项目选题推荐
  • (十六)一篇文章学会Java的常用API
  • (转)项目管理杂谈-我所期望的新人
  • .cfg\.dat\.mak(持续补充)
  • .FileZilla的使用和主动模式被动模式介绍
  • .NET Framework 和 .NET Core 在默认情况下垃圾回收(GC)机制的不同(局部变量部分)
  • .net 开发怎么实现前后端分离_前后端分离:分离式开发和一体式发布
  • .NET 药厂业务系统 CPU爆高分析
  • .NET(C#) Internals: as a developer, .net framework in my eyes
  • .NET/C# 推荐一个我设计的缓存类型(适合缓存反射等耗性能的操作,附用法)
  • .NET高级面试指南专题十一【 设计模式介绍,为什么要用设计模式】
  • .net获取当前url各种属性(文件名、参数、域名 等)的方法
  • .NET使用HttpClient以multipart/form-data形式post上传文件及其相关参数
  • //解决validator验证插件多个name相同只验证第一的问题
  • @private @protected @public
  • @RequestParam @RequestBody @PathVariable 等参数绑定注解详解
  • @staticmethod和@classmethod的作用与区别
  • [2013][note]通过石墨烯调谐用于开关、传感的动态可重构Fano超——