英文注釋

大意如下:
一個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 方法。

在RecyclerViewPool 中 :




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 的最大尺寸。

總結(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)于本人描述如有不確切之地方,還請大家予以指正,感謝。