項目用到ListView,由於要用到ImageView,圖片源不是在資源裡面的,沒法使用資源ID,因此無法直接使用SimpleAdapter,要自己寫一個Adapter。在使用ListView和Adapter需要注意以下幾點:
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來驗證這個過程:
先看關鍵代碼:
Java代碼
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在螢幕中的最大高度所引起。希望有知道的朋友能夠告訴,先謝謝了!
最後,如果上面有什麼地方說錯的話,希望能夠指出,互相進步嘛。