android開發中ListView與Adapter使用要點介紹

來源:互聯網
上載者:User

1. Adapter.getView()

public View getView(int position, View convertView , ViewGroup parent){...}

這個方法就是用來獲得指定位置要顯示的View。官網解釋如下:
Get a View that displays the data at the specified position in the data set. You can either create a View manually or inflate it from an XML layout file.

當要顯示一個View就調用一次這個方法。這個方法是ListView效能好壞的關鍵。方法中有個convertView,這個是Android在為我們而做的緩衝機制。
ListView中每個item都是通過getView返回並顯示的,假如item有很多個,那麼重複建立這麼多個物件來顯示顯然是不合理。因此,Android提供了Recycler,將沒有正在顯示的item放進RecycleBin,然後在顯示新視圖時從RecycleBin中複用這個View。

Recycler的工作原理大致如下:
假設螢幕最多能看到11個item,那麼當第1個item滾出螢幕,這個item的View進入RecycleBin中,第12個要出現前,通過getView從資源回收筒(RecycleBin)中重用這個View,然後設定資料,而不必重新建立一個View。

我們用Android提供的APIDemos來驗證這個過程:

先看關鍵代碼: 複製代碼 代碼如下:public View getView(int position, View convertView, ViewGroup parent) {
// A ViewHolder keeps references to children views to avoid unneccessary calls
// to findViewById() on each row.
ViewHolder holder;
// When convertView is not null, we can reuse it directly, there is no need
// to reinflate it. We only inflate a new View when the convertView supplied
// by ListView is null.
if (convertView == null) {
convertView = mInflater.inflate(R.layout.list_item_icon_text, null);
Log.v("tag", "positon " + position + " convertView is null, " + "new: " + convertView);
// Creates a ViewHolder and store references to the two children views
// we want to bind data to.
holder = new ViewHolder();
holder.text = (TextView) convertView.findViewById(R.id.text);
holder.icon = (ImageView) convertView.findViewById(R.id.icon);
convertView.setTag(holder);
} else {
// Get the ViewHolder back to get fast access to the TextView
// and the ImageView.
holder = (ViewHolder) convertView.getTag();
Log.v("tag", "positon " + position + " convertView is not null, " + convertView);
}
// Bind the data efficiently with the holder.
holder.text.setText(DATA[position]);
holder.icon.setImageBitmap((position & 1) == 1 ? mIcon1 : mIcon2);
return convertView;
}

static class ViewHolder {
TextView text;
ImageView icon;
}

可以看到,一開啟Activity,看到10個item.

我們看看Log資訊:

可以看出,每次convertView都是null, 都是建立一個View來顯示的。

當我們向下滑動,如,

由於item0和item10都顯示一半,所以item10也是建立出來,但是當要顯示item11的時候,由於item0已經不在螢幕上,所以item11複用了item0的執行個體。可以從以下Log資訊看出:

我們分析Log資訊,可以看出item11的對象是item0, item12的對象是item1,如此類推。

這樣,通過複用convertView,就可以避免每次都建立View,節省記憶體而且最佳化ListView的滑動效果。

2. ListView的Layout XML


除了上述說的,還有一個要點就是ListView在Layout XML中的描述。

先看問題:

有時,我們可能會看到一開啟ListView,getView會重複調用好次(假設螢幕最多可以看到6個item),如:

一直重複 0-6, 0-5,0-5, 0-5,0-5, 0-5。而且,convertView一開始都是同一個View,這個是因為ListView的

android:layout_height="wrap_content"。

我們修改為android:layout_height="fill_parent",Log資訊如下:

可以看出,修改之後ListView的getView調用恢複和Recycler的行為一致。

至於為什麼使用wrap_content會出現重複調用的情況,我還沒有研究過。不過初步覺得是因為在Android描繪ListView的時候,由於不清楚高度,所以使用一個item去試探ListView在螢幕中的最大高度所引起。希望有知道的朋友能夠告訴,先謝謝了!

最後,如果上面有什麼地方說錯的話,希望能夠指出,互相進步嘛。

補充:

在接著使用ListView的時候,又發現一個很奇怪的現象。調用notifyDataSetChanged()之後,ListView在重新getView()時,所有的convertView的順序都逆序了。請看下面:

這應該是由於recycleBin是stack結構而引起。

其它:

1. Disable divider:

android:divider="#00000000"
android:dividerHeight="0dp"

2. Disable ListView selector:

convertView.setOnClickListener(null);
如果只是要去掉顏色,可以用android:listSelector="#00000000"

3. Disable header divider:

android:headerDividersEnabled="false"
4. getItemViewType(int)與getItemViewType(int)

getItemViewType(int) can not return int value larger than getViewTypeCount().
Otherwise you will get java.lang.ArrayIndexOutOfBoundsException at android.widget.AbsListView$RecycleBin.addScrapView(AbsListView.java:3523)
ListView會根據不同的ViewType返回相應type的convertView.

一般寫法:

複製代碼 代碼如下:getView() {
switch (getItemViewType(position)) {
case type1:
if(convertView == null) {
} else {
}
break;
case type2:
default:
if(convertView == null) {
} else {
}
break;
}
return convertView;
}
getItemViewType(int position) {
// 根據情境,一般有:
// 1. 不同的item type對應的position是固定的,那麼ListView的data可以分別存放
// 2. 不同的item type對應的position是不固定的,那麼可以把ListView的data統一放在List<Object>中,
// 然後使用instanceof來判斷Object的類型進而區分position對應的view type.
}
相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.