雖然現在5.0後Google推出了RecycleView,但在5.0 Lollipop普及前Listview仍會被廣泛使用,所以打算再次探究一下Listview的源碼,了解一下Listview 的構成及加載機制。
上圖簡單梳理了Listview的構成及與其相關類之間的關系,並簡要地列出了些重要的方法和內部類。
從上圖可以清晰的看出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是繼承自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一樣,找機會要研究一下。
更多Android相關信息見Android 專題頁面 http://www.linuxidc.com/topicnews.aspx?tid=11