这是RecyclerView
缓存机制系列文章的第四篇,系列文章的目录以下:缓存
第一篇中遗留的一个问题尚未解决:复用表项时优先级最高的scrap view
是用来干吗的?这篇文章试着经过阅读源码来解答这个问题。bash
scrap view
对应的存储结构是final ArrayList<ViewHolder> mAttachedScrap = new ArrayList<>();
。理解成员变量用途的最好办法是 “搜索它在何时被访问” 。对于列表结构来讲就至关于 1. 在何时往列表添加内容? 2. 在何时清空列表内容?app
全局搜索mAttachedScrap
被访问的地方,其中只有一处调用了mAttachedScrap.add()
:oop
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);
}
}
}
复制代码
沿着调用链继续往上:布局
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);
}
}
}
复制代码
根据viewHolder
的不一样状态,要么将其添加到mAttachedScrap
集合,要么将其存入回收池。其中recycleViewHolderInternal()
在RecyclerView缓存机制(回收去哪?)分析过。 沿着调用链继续向上:post
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
中,回收数据的来源是LayoutManager
的孩子,而LayoutManager
的孩子都是屏幕上可见的表项。mAttachedScrap
用于屏幕中可见表项的回收和复用全局搜索mAttachedScrap
被访问的地方,其中只有一处调用了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状态
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();
}
}
}
复制代码
沿着调用链向上:ui
public class RecyclerView extends ViewGroup implements ScrollingView, NestedScrollingChild2 {
//RecyclerView布局的最后一步
private void dispatchLayoutStep3() {
...
mLayout.removeAndRecycleScrapInt(mRecycler);
...
}
复制代码
至此能够得出结论:mAttachedScrap
生命周期起始于RecyclerView
布局开始,终止于RecyclerView
布局结束。this
通过四篇文章的分析,RecyclerVeiw
的四级缓存都分析完了,总结以下:spa
Recycler
有4个层次用于缓存ViewHolder
对象,优先级从高到底依次为ArrayList<ViewHolder> mAttachedScrap
、ArrayList<ViewHolder> mCachedViews
、ViewCacheExtension mViewCacheExtension
、RecycledViewPool mRecyclerPool
。若是四层缓存都未命中,则从新建立并绑定ViewHolder
对象
缓存性能:
缓存 | 从新建立ViewHolder |
从新绑定数据 |
---|---|---|
mAttachedScrap | false | false |
mCachedViews | false | false |
mRecyclerPool | false | true |
缓存容量:
mAttachedScrap
:没有大小限制,但最多包含屏幕可见表项。mCachedViews
:默认大小限制为2,放不下时,按照先进先出原则将最早进入的ViewHolder
存入回收池以腾出空间。mRecyclerPool
:对ViewHolder
按viewType
分类存储(经过SparseArray
),同类ViewHolder
存储在默认大小为5的ArrayList
中。缓存用途:
mAttachedScrap
:用于布局过程当中屏幕可见表项的回收和复用。mCachedViews
:用于移出屏幕表项的回收和复用,且只能用于指定位置的表项,有点像“回收池预备队列”,即老是先回收到mCachedViews
,当它放不下的时候,按照先进先出原则将最早进入的ViewHolder
存入回收池。mRecyclerPool
:用于移出屏幕表项的回收和复用,且只能用于指定viewType
的表项缓存结构:
mAttachedScrap
:ArrayList<ViewHolder>
mCachedViews
:ArrayList<ViewHolder>
mRecyclerPool
:对ViewHolder
按viewType
分类存储在SparseArray<ScrapData>
中,同类ViewHolder
存储在ScrapData
中的ArrayList
中