再次探究Android ListView快取機制

XycZero的部落格發表於2015-01-22

概述

雖然現在5.0後Google推出了RecycleView,但在5.0 Lollipop普及前Listview仍會被廣泛使用,所以打算再次探究一下Listview的原始碼,瞭解一下Listview 的構成及載入機制。

探究

enter image description here

上圖簡單梳理了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一樣,找機會要研究一下。

相關文章