RecyclerView緩存機制(scrap view)

這是RecyclerView緩存機制系列文章的第四篇材蛛,系列文章的目錄如下:

  1. RecyclerView緩存機制(咋復(fù)用深碱?)
  2. RecyclerView緩存機制(回收些啥?)
  3. RecyclerView緩存機制(回收去哪?)
  4. RecyclerView緩存機制(scrap view)

第一篇中遺留的一個問題還沒有解決:復(fù)用表項時優(yōu)先級最高的scrap view是用來干嘛的?這篇文章試著通過閱讀源碼來解答這個問題毅整。

scrap view對應(yīng)的存儲結(jié)構(gòu)是final ArrayList<ViewHolder> mAttachedScrap = new ArrayList<>();。理解成員變量用途的最好辦法是 “搜索它在什么時候被訪問” 绽左。對于列表結(jié)構(gòu)來說就相當于 1. 在什么時候往列表添加內(nèi)容悼嫉? 2. 在什么時候清空列表內(nèi)容?

添加內(nèi)容

全局搜索mAttachedScrap被訪問的地方拼窥,其中只有一處調(diào)用了mAttachedScrap.add():

public final class Recycler {
        /**
         * Mark an attached view as scrap.
         * 回收ViewHolder到scrap集合(mAttachedScrap或mChangedScrap)
         *
         * <p>"Scrap" views are still attached to their parent RecyclerView but are eligible
         * for rebinding and reuse. Requests for a view for a given position may return a
         * reused or rebound scrap view instance.</p>
         * scrap view依然依附于它的父親戏蔑。。鲁纠。
         *
         * @param view View to scrap
         */
        void scrapView(View view) {
            final ViewHolder holder = getChildViewHolderInt(view);
            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."
                            + " Invalid views cannot be reused from scrap, they should rebound from"
                            + " recycler pool." + exceptionLabel());
                }
                holder.setScrapContainer(this, false);
                //添加到mAttachedScrap集合中
                mAttachedScrap.add(holder);
            } else {
                if (mChangedScrap == null) {
                    mChangedScrap = new ArrayList<ViewHolder>();
                }
                holder.setScrapContainer(this, true);
                //添加到mChangedScrap集合中
                mChangedScrap.add(holder);
            }
        }
}

沿著調(diào)用鏈繼續(xù)往上:

public abstract static class LayoutManager {
        private void scrapOrRecycleView(Recycler recycler, int index, View view) {
            final ViewHolder viewHolder = getChildViewHolderInt(view);
            if (viewHolder.shouldIgnore()) {
                if (DEBUG) {
                    Log.d(TAG, "ignoring view " + viewHolder);
                }
                return;
            }
            //刪除表項并入回收池
            if (viewHolder.isInvalid() && !viewHolder.isRemoved()
                    && !mRecyclerView.mAdapter.hasStableIds()) {
                removeViewAt(index);
                recycler.recycleViewHolderInternal(viewHolder);
            }
            //detach表項并入scrap集合
            else {
                detachViewAt(index);
                recycler.scrapView(view);
                mRecyclerView.mViewInfoStore.onViewDetached(viewHolder);
            }
        }
}

根據(jù)viewHolder的不同狀態(tài)总棵,要么將其將其添加到mAttachedScrap集合,要么將其存入回收池改含。其中recycleViewHolderInternal()RecyclerView緩存機制(回收去哪情龄?)分析過。
沿著調(diào)用鏈繼續(xù)向上:

public abstract static class LayoutManager {
        /**
         * Temporarily detach and scrap all currently attached child views. Views will be scrapped
         * into the given Recycler. The Recycler may prefer to reuse scrap views before
         * other views that were previously recycled.
         * 暫時將當可見表項進行分離并回收
         *
         * @param recycler Recycler to scrap views into
         */
        public void detachAndScrapAttachedViews(Recycler recycler) {
            final int childCount = getChildCount();
            //遍歷所有可見表項并回收他們
            for (int i = childCount - 1; i >= 0; i--) {
                final View v = getChildAt(i);
                scrapOrRecycleView(recycler, i, v);
            }
        }
       /**
         * Lay out all relevant child views from the given adapter.
         * 從給定的adapter布局所有的孩子
         */
        public void onLayoutChildren(Recycler recycler, State state) {
            ...
            //在填充表項之前回收所有表項
            detachAndScrapAttachedViews(recycler);
            ...
            if (mAnchorInfo.mLayoutFromEnd) {
                ...
                //填充表項
                fill(recycler, mLayoutState, state, false);
                ...
            }
            ...
        }
}

public class RecyclerView extends ViewGroup implements ScrollingView, NestedScrollingChild2 {
    //RecyclerView布局的第二步
    private void dispatchLayoutStep2() {
        ...
        mLayout.onLayoutChildren(mRecycler, mState);
        ...
    }
}
  • 在將表項一個個填充到列表之前會先將其先回收到mAttachedScrap中候味,回收數(shù)據(jù)的來源是LayoutManager的孩子,而LayoutManager的孩子都是屏幕上可見的表項隔心。
  • 注釋中“暫時將當可見表項進行分離并回收”白群,既然是“暫時回收”,那待會必然會發(fā)生“復(fù)用”硬霍。復(fù)用邏輯可移步RecyclerView緩存機制(咋復(fù)用帜慢?)
  • 至此可以得出結(jié)論:mAttachedScrap用于屏幕中可見表項的回收和復(fù)用

清空內(nèi)容

全局搜索mAttachedScrap被訪問的地方,其中只有一處調(diào)用了mAttachedScrap.clear():

public final class Recycler {
        void clearScrap() {
            mAttachedScrap.clear();
            if (mChangedScrap != null) {
                mChangedScrap.clear();
            }
        }
}

public abstract static class LayoutManager {
        /**
         * Recycles the scrapped views.
         * 回收所有scrapped view
         */
        void removeAndRecycleScrapInt(Recycler recycler) {
            final int scrapCount = recycler.getScrapCount();
            // Loop backward, recycler might be changed by removeDetachedView()
            //遍歷搜有scrap view并重置ViewHolder狀態(tài)
            for (int i = scrapCount - 1; i >= 0; i--) {
                final View scrap = recycler.getScrapViewAt(i);
                final ViewHolder vh = getChildViewHolderInt(scrap);
                if (vh.shouldIgnore()) {
                    continue;
                }
                vh.setIsRecyclable(false);
                if (vh.isTmpDetached()) {
                    mRecyclerView.removeDetachedView(scrap, false);
                }
                if (mRecyclerView.mItemAnimator != null) {
                    mRecyclerView.mItemAnimator.endAnimation(vh);
                }
                vh.setIsRecyclable(true);
                recycler.quickRecycleScrapView(scrap);
            }
            //清空scrap view集合
            recycler.clearScrap();
            if (scrapCount > 0) {
                mRecyclerView.invalidate();
            }
        }
}

沿著調(diào)用鏈向上:

public class RecyclerView extends ViewGroup implements ScrollingView, NestedScrollingChild2 {
    //RecyclerView布局的最后一步
    private void dispatchLayoutStep3() {
        ...
        mLayout.removeAndRecycleScrapInt(mRecycler);
        ...
}

至此可以得出結(jié)論:mAttachedScrap生命周期起始于RecyclerView布局開始唯卖,終止于RecyclerView布局結(jié)束粱玲。

總結(jié)

經(jīng)過四篇文章的分析,RecyclerVeiw的四級緩存都分析完了拜轨,總結(jié)如下:

  1. Recycler有4個層次用于緩存ViewHolder對象抽减,優(yōu)先級從高到底依次為ArrayList<ViewHolder> mAttachedScrapArrayList<ViewHolder> mCachedViews橄碾、ViewCacheExtension mViewCacheExtension卵沉、RecycledViewPool mRecyclerPool。如果四層緩存都未命中法牲,則重新創(chuàng)建并綁定ViewHolder對象

  2. 緩存性能:

    緩存 重新創(chuàng)建ViewHolder 重新綁定數(shù)據(jù)
    mAttachedScrap false false
    mCachedViews false false
    mRecyclerPool false true
  1. 緩存容量:
    • mAttachedScrap:沒有大小限制史汗,但最多包含屏幕可見表項。
    • mCachedViews:默認大小限制為2拒垃,放不下時停撞,按照先進先出原則將最先進入的ViewHolder存入回收池以騰出空間。
    • mRecyclerPool:對ViewHolderviewType分類存儲(通過SparseArray)悼瓮,同類ViewHolder存儲在默認大小為5的ArrayList中戈毒。
  2. 緩存用途:
    • mAttachedScrap:用于布局過程中屏幕可見表項的回收和復(fù)用艰猬。
    • mCachedViews:用于移出屏幕表項的回收和復(fù)用,且只能用于指定位置的表項副硅,有點像“回收池預(yù)備隊列”姥宝,即總是先回收到mCachedViews,當它放不下的時候恐疲,按照先進先出原則將最先進入的ViewHolder存入回收池腊满。
    • mRecyclerPool:用于移出屏幕表項的回收和復(fù)用,且只能用于指定viewType的表項
  3. 緩存結(jié)構(gòu):
    • mAttachedScrapArrayList<ViewHolder>
    • mCachedViewsArrayList<ViewHolder>
    • mRecyclerPool:對ViewHolderviewType分類存儲在SparseArray<ScrapData>中培己,同類ViewHolder存儲在ScrapData中的ArrayList
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末碳蛋,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子省咨,更是在濱河造成了極大的恐慌肃弟,老刑警劉巖,帶你破解...
    沈念sama閱讀 207,113評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件零蓉,死亡現(xiàn)場離奇詭異笤受,居然都是意外死亡,警方通過查閱死者的電腦和手機敌蜂,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,644評論 2 381
  • 文/潘曉璐 我一進店門箩兽,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人章喉,你說我怎么就攤上這事汗贫。” “怎么了秸脱?”我有些...
    開封第一講書人閱讀 153,340評論 0 344
  • 文/不壞的土叔 我叫張陵落包,是天一觀的道長。 經(jīng)常有香客問我摊唇,道長咐蝇,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,449評論 1 279
  • 正文 為了忘掉前任巷查,我火速辦了婚禮嘹害,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘吮便。我一直安慰自己笔呀,他們只是感情好,可當我...
    茶點故事閱讀 64,445評論 5 374
  • 文/花漫 我一把揭開白布髓需。 她就那樣靜靜地躺著许师,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上微渠,一...
    開封第一講書人閱讀 49,166評論 1 284
  • 那天搭幻,我揣著相機與錄音,去河邊找鬼逞盆。 笑死檀蹋,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的云芦。 我是一名探鬼主播俯逾,決...
    沈念sama閱讀 38,442評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼舅逸!你這毒婦竟也來了桌肴?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,105評論 0 261
  • 序言:老撾萬榮一對情侶失蹤琉历,失蹤者是張志新(化名)和其女友劉穎坠七,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體旗笔,經(jīng)...
    沈念sama閱讀 43,601評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡彪置,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,066評論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了蝇恶。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片拳魁。...
    茶點故事閱讀 38,161評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖艘包,靈堂內(nèi)的尸體忽然破棺而出的猛,到底是詐尸還是另有隱情耀盗,我是刑警寧澤想虎,帶...
    沈念sama閱讀 33,792評論 4 323
  • 正文 年R本政府宣布,位于F島的核電站叛拷,受9級特大地震影響舌厨,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜忿薇,卻給世界環(huán)境...
    茶點故事閱讀 39,351評論 3 307
  • 文/蒙蒙 一裙椭、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧署浩,春花似錦揉燃、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,352評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春抢腐,著一層夾襖步出監(jiān)牢的瞬間姑曙,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,584評論 1 261
  • 我被黑心中介騙來泰國打工迈倍, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留伤靠,地道東北人。 一個月前我還...
    沈念sama閱讀 45,618評論 2 355
  • 正文 我出身青樓啼染,卻偏偏與公主長得像宴合,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子提完,可洞房花燭夜當晚...
    茶點故事閱讀 42,916評論 2 344