RecyclerView笔记

参考
基于滑动场景解析RecyclerView的回收复用机制原理
RecyclerView剖析

一、 缓存机制
1.1 scrap缓存

// 添加
 LayoutManager#onLayoutChildren
       // 删除recyclerview中的子view并保存到scrap
       detachAndScrapAttachedViews(recycler);
       // 从4级缓存中查找viewholder后执行addView
// 删除
 RecyclerView#dispatchLayoutStep3
      // 重置scrap,即清空
      mLayout.removeAndRecycleScrapInt(mRecycler);     

提示:mAttachedScrap和mChangedScrap只用在layout过程中的临时变量, 当recyclerview布局完成后(即STATE_IDLE后)scrap是空数组;

1.2 mCachedViews
添加到RecyclerView时不用执行onBindViewHolder函数, 默认大小是2; 从安卓5开始添加了Prefetch功能, 如果使用LinearLayoutManager则大小增1; 如果使用瀑布流StaggeredGridLayoutManager则增加span的个数(例如2列瀑布流则增2)。
mCachedView数量 = mRequestedCacheMax + prefetch数量

在做滑动性能调优时可以增加该数组长度, 减少onBindViewHolder的执行次数;

二、瀑布流
瀑布流最为复杂, 坑也最多。
常见顶部留白和item抖动是由于viewholder在layout时和实际高度有差别;
(百度能搜到一堆,不再赘述)

1、 在onBindViewHolder中设置itemView的宽度和高度为固定值;
2、 设置GAP_HANDLING_NONE
3、 取消默认动画, RecyclerView#setItemAnimator(null);
4、 滑动到顶部时执行invalidateSpanAssignments, 其实就是清空mLazyLookupSpan并requestLayout。

三、布局流程
RecyclerView#setAdapter
RecyclerView.Adapter#notifyDataSetChanged
实际上都是修改标志位,然后执行requestLayout()。 等到下一个Choreographer触发的doTraversal函数;

dispatchLayoutStep1:  preLayout阶段, 会缓存动画信息到mViewInfoStore并可能执行一次mLayout.onLayoutChildren
dispatchLayoutStep2: 真正的布局阶段, 肯定执行mLayout.onLayoutChildren函数;
dispatchLayoutStep3:postLayout阶段,执行mViewInfoStore中的动画并清空scrap数组, 执行mLayout.onLayoutComplete

tip:重载LayoutManager#onLayoutComplete函数表示布局已完成;

四、瀑布流缓存
mLazySpanLookup保存每个item在第几列, 从而滑动列表后记录以前的位置并在重新显示时直接使用。否则使用getNextSpan函数计算出在哪一列;
mSpans数组, 元素个数等于列数。 保存每一列的view和开始、结束座标, 从而在每一列添加view时从mSpans中拿到以前的位置并添加。

mSpans是瀑布流的大坑, 这意味着notifyDataSetChanged后并不会从顶部开始添加view, 而是从mSpans缓存的位置添加。
下拉刷新时添加滑动位置, 其实就是设置参数并在一次layout中完成操作;

notifyDateSetChanged
RecyclerView#scrollToPosition(0);    


                // Child is not visible. Set anchor coordinate depending on in which direction
                // child will be visible.
                anchorInfo.mPosition = mPendingScrollPosition;
                if (mPendingScrollPositionOffset == INVALID_OFFSET) {
                    final int position = calculateScrollDirectionForPosition(
                            anchorInfo.mPosition);
                    anchorInfo.mLayoutFromEnd = position == LayoutState.LAYOUT_END;
                    anchorInfo.assignCoordinateFromPadding();
                } else {
                    anchorInfo.assignCoordinateFromPadding(mPendingScrollPositionOffset);
                }
                anchorInfo.mInvalidateOffsets = true;
    // 执行scrollToPosition(0)后mInvalidateOffsets为true, 并重置了mSpan
    if (getChildCount() > 0 && (mPendingSavedState == null
            || mPendingSavedState.mSpanOffsetsSize < 1)) {
        if (anchorInfo.mInvalidateOffsets) {
            for (int i = 0; i < mSpanCount; i++) {
                // Scroll to position is set, clear.
                mSpans[i].clear();
                if (anchorInfo.mOffset != INVALID_OFFSET) {
                    mSpans[i].setLine(anchorInfo.mOffset);
                }
            }

五、神坑
瀑布流放开item动画后出现诡异问题如被删除的item再次显示到recyclerview(上下滑动后刷新为正确的数据)、间距各种不对。
解决措施:在onScrollListener里的statechanged函数里判断当前正在itemanimation动画则结束动画;

常见问题:
1、滑动列表时先删除再添加view;
2、recyclerview使用观察者模式, 更新各种标志位后调用requestLayout; requestLayout函数可以调用多次, 其实就是设置标志位;
3、瀑布流滑动一段距离后执行notifyDataSetChanged时按照mSpans的位置进行layout。
4、itemdecoration是在RecyclerView的onDraw函数中绘制的。
后续抽时间补上流程图、示意图。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,133评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,682评论 3 390
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,784评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,508评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,603评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,607评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,604评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,359评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,805评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,121评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,280评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,959评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,588评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,206评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,442评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,193评论 2 367
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,144评论 2 352