RecyclerView 之 Recycler & RecycledViewPool 源碼解析

英文注釋

大意如下:

一個Recycler 負(fù)責(zé)管理 廢棄的和獨(dú)立的 itemview 用于重用。

一個"scrapped" (意為廢棄的) View 是一個仍然附屬在父RecyclerView 但是已經(jīng)被標(biāo)記為可移除或重用。

如果這個被重用的View被認(rèn)為是臟數(shù)據(jù),adapter 將會被調(diào)用并重新綁定它,反之,這個view 可以在沒有任何前置工作的情況下快速被LayoutManager 重用。

本人認(rèn)為此處臟數(shù)據(jù)的View 意為:緩存在內(nèi)存中,暫沒被adapter 綁定的數(shù)據(jù)。




Recycler 部分源碼:


回收部分:

recycleViewHolderInternal ?是一個內(nèi)部實(shí)現(xiàn)的方法檢查view 是不是廢棄的或者附屬在RecycleView 上,并拋出對應(yīng)的異常,同時該方法會調(diào)用RecycledViewPool。

已經(jīng)標(biāo)記為廢棄的,或者父view 不是空 也就說此時ItemView仍然附屬在RecyclerView上,不能被回收,所以拋出異常。

recycleViewHolderInternal 方法中三個布爾值:

forceRecycle:是否可以強(qiáng)制回收

recycled :是否已經(jīng)回收

cached : 默認(rèn)為false 即 尚未被緩存


如果ViewHolder 被系統(tǒng)認(rèn)為要強(qiáng)制回收 或者 holder 是可以被回收

? ? 如果被緩存的數(shù)量 等于 最大緩存數(shù)(mViewCacheMax 默認(rèn)為2) 且 緩存數(shù)>0


step1:


Recycler 中最大緩存數(shù)是2 ,被回收一個 還余1 ?所以下面條件一定成立,

mCachedViews ?中加入新的viewholder ,即:參數(shù)中所傳ViewHolder 被緩存到Recycle。


step2:



step2 執(zhí)行后 cached 必然為 true. ?則不會執(zhí)行下面

addViewHolderToRecycledViewPool 方法。 反而在 setp1 中會執(zhí)行 recycleCachedViewAt(0);


如step1 step2 均不能執(zhí)行則執(zhí)行:

step3:


看看 step1 中 recycleCachedViewAt(pos)

{...... ?

addViewHolderToRecycledViewPool(viewHolder);

mCachedViews.remove(cachedViewIndex);

}

把viewholder 加入到RecyclerViewPool 中,同時從Recycler 中移除。


Recycler 緩存 總結(jié):Recycler 提供一個 最大尺寸為 2 的ViewHolder緩存 ,

當(dāng)緩存尺寸<2的 時候,直接將可以被回收或可以強(qiáng)制回收的ViewHolder 存入該緩存;

當(dāng)緩存尺寸 == 2 的時候,如果還要存入新的ViewHolder ,則從該緩存中移除最先加入的ViewHolder(pos -->0),且 將其存入RecyclerViewPool 中,同時將新ViewHolder(pos --> cacheSize-1) 存入該緩存(Recycler 緩存)。





看看RecyclerViewPool:


RecycledViewPool 可以讓你在多個不同的RecyclerViews 之間 共享 itemViews,

使用RecyclerView 提供的 setRecycledViewPool 方法。

如果你不手動提供一個緩存池,RecyclerView 會自動創(chuàng)建一個 pool 供它自己使用。




RecyclerView所提供方法setRecycledViewPool

getRecycledViewPool

都是通過Recycler 調(diào)用。


在Recycler 中 聲明了

private RecycledViewPool mRecyclerPool;

RecycledViewPool getRecycledViewPool() {

? ? if(mRecyclerPool==null) {

? ? ? ? ?mRecyclerPool=new ? ? ? ? ? ? ? ? ?RecycledViewPool();

? ? }

? ? ?returnmRecyclerPool;

}

由此可見 ?RecycledViewPool 并非隨著Recycler 的初始化而初始化,而是在調(diào)用時才初始化。

同時Recycler 提供setRecycledViewPool 方法:


void setRecycledViewPool(RecycledViewPool pool) {

if(mRecyclerPool!=null) {

? ? ? mRecyclerPool.detach();

}

mRecyclerPool= pool;

? if(pool !=null) {

? ? ? ?mRecyclerPool.attach(getAdapter());

? }

}


看到上述2個方法可以想象下,我是不是可以這樣玩:

RecyclerView.RecycledViewPool pool =newRecyclerView.RecycledViewPool();

setRecycledViewPool(pool);

這樣多個recyclerView 可以共用同一個pool

但是pool本身可以由Recycle調(diào)用getRecycledViewPool創(chuàng)建,而且RecyclerView提供了可供外部調(diào)用的方法:

RecycledViewPool getRecycledViewPool() {

? return mRecycler.getRecycledViewPool();

}

其實(shí)就沒必要自己去new 一個 RecycledViewPool 了,直接如此即可:

recyclerView2.setRecycledViewPool(recyclerView1.getRecycledViewPool());




RecycledViewPool 部分代碼:


在Recycler 中 有方法addViewHolderToRecyclerViewPool, 圖1可見,調(diào)用了RecyclerViewPool 的 putRecycledView 方法。

圖 1


在RecyclerViewPool 中 :

圖 2


圖 3


圖 4


圖 5

mScrap 是一個維持 ArrayList<ViewHolder> ?類型的優(yōu)化過的鍵值對數(shù)組SparseArray;

SparseArray 可以 代替 Map 使用,比Map 性能好。


圖 4 所示: mScrap.put(viewType, scrap);

mScrap 鍵是 viewType,不同的viewTpe 意味著不同的 ViewHolder,在抽象編程中,為了統(tǒng)一管理,即使不看源代碼 也可知 ArrayList<ViewHolder> 中的泛型 ViewHolder 必然是 抽象類。


mScrap 值 是 ArrayList<ViewHolder> ,如果為空 則創(chuàng)建一個,并且 利用

SparseIntArray mMaxScrap ,維持最大尺寸為 5。



代碼分析到這里就能明白:


RecycledViewPool ? 緩存的不同 類型viewType 的數(shù)量是不限的,但是每個viewType 的具體ViewHolder 最多為5個??梢酝ㄟ^圖 6 所示方法 設(shè)置某個viewType 的最大尺寸。


圖 6







總結(jié)(在這里暫不考慮 ?ViewCacheExtension):


在RecyclerView 中,ViewHolder 存在的位置,經(jīng)過了三次變化:


一是Adapter 綁定 且 可見,此時 ViewHolder 是不能被系統(tǒng)強(qiáng)制回收,也不允許程序員手動回收;


二是在Recycler 中,此時ViewHolder 狀態(tài)可以是被標(biāo)記狀態(tài)(可從RecyclerView移除,也可以被重用),或者是處于獨(dú)立狀態(tài)(本人理解是獨(dú)立于RecycleView,不可見狀態(tài),但是仍然和Adapter關(guān)聯(lián));


三是在RecycleViewPool 中,此時ViewHolder 完全是廢棄的狀態(tài),除非被再次使用,否則直到被系統(tǒng)徹底回收。


關(guān)于本人描述如有不確切之地方,還請大家予以指正,感謝。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容