探究Android ListView緩存機制
概述
雖然現在5.0后Google推出了RecycleView,但在5.0 Lollipop普及前Listview仍會被廣泛使用,所以打算再次探究一下Listview的源碼,了解一下Listview 的構成及加載機制。
探究

上圖簡單梳理了Listview的構成及與其相關類之間的關系,并簡要地列出了些重要的方法和內部類。
AdapterView
從上圖可以清晰的看出Listview歸根究底是繼承自AdapterView。AdaterView是一個抽象類,一些最基本和通用方法或接口都是在此定義或聲明的,其中一些更是開發者所常用的,諸如:
//Item Click 監聽接口 /** * Interface definition for a callback to be invoked when an item in this * AdapterView has been clicked. */ public interface OnItemClickListener { ... ... void onItemClick(AdapterView<?> parent, View view, int position, long id); } //設置Adapter抽象方法 /** * Sets the adapter that provides the data and the views to represent the data * in this widget. * * @param adapter The adapter to use to create this view's content. */ public abstract void setAdapter(T adapter);
此外在AdapterView中實現了DataSetObserver抽象類,我們一般調用mAdapter.notifyChanged()所觸發的就是DataSetObserver的onChanged()方法。關鍵源碼如下:
class AdapterDataSetObserver extends DataSetObserver {
private Parcelable mInstanceState = null;
@Override
public void onChanged() {
mDataChanged = true;
mOldItemCount = mItemCount;
mItemCount = getAdapter().getCount();
... ...
}
@Override
public void onInvalidated() {
mDataChanged = true;
... ...
}
... ...
}AbsListView
AbsListView是繼承自AdapterView,在該類中實現了一個非常重要的內部類RecycleBin,內部類RecycleBin其 實就是AbsListView緩存機制的核心類,它的作用是管理AbsListView的item存儲和取得。AbsListview的緩存分為兩級,第 一級為activeView,第二級為scrapview。二者的間的轉換主要是在layoutChildren()方法進行(該抽象方法在 LisView中實現),具體分析見如下源碼:
@Override
protected void layoutChildren() {
... ...
//說明RecycleBin并不緩存HeadView和FooterView
// Don't put header or footer views into the Recycler.
//Those are already cached in mHeaderViews;
if (dataChanged) {
//如果data改變了,則當前所有childView都添加至mScrapViews;
for (int i = 0; i < childCount; i++) {
recycleBin.addScrapView(getChildAt(i), firstPosition+i);
if (ViewDebug.TRACE_RECYCLER) {
ViewDebug.trace(getChildAt(i),
ViewDebug.RecyclerTraceType.MOVE_TO_SCRAP_HEAP, index, i);
}
}
} else {
//若data未改變,即第一次加載時,根據當前childCount數量對mArchiveViews賦值。
recycleBin.fillActiveViews(childCount, firstPosition);
}
... ...
switch (mLayoutMode) {
... ...(在switch條件中執行makeAndAddView函數)
}
// Flush any cached views that did not get reused above
//執行makeAndAddView函數后將需要顯示的item view已添加至ListView中,
//所以跳出siwtch后會將緩存的mActiveViews全部轉換為mScrapViews。
recycleBin.scrapActiveViews();
... ...
}同時AbsListview中定義了一個ObtainView方法,一般地當Listview加載時若發現沒有可復用的itemView時要么從 RecycleBin中轉換ScrapView都要么是通過mAdapter.getView()獲取新的itemView,ObtainView方法就 是專門用來處理上述的兩種情況,具體分析如下:
View obtainView(int position, boolean[] isScrap) {
... ...
scrapView = mRecycler.getScrapView(position);
View child;
//若scrapView不為空,則將scrapView轉換為可復用的itemView
if (scrapView != null) {
... ...
child = mAdapter.getView(position, scrapView, this);
... ...
}else{
//若scrapView為空,則通過adapter.getView()函數獲取新的ItemView
child = mAdapter.getView(position, null, this);
... ...
}
}結語
OK,今天就先總結這么多了,不足之處歡迎指出。當然今后使用RecycleView會是一種趨勢,和AS一樣,找機會要研究一下。
作者:XycZero
查看原文:http://www.xyczero.com/blog/article/18/.