listview加載性能優化ViewHolder (轉)

在android開發中Listview是一個很重要的組件,它以列表的形式根據數據的長自適應展示具體內容,用戶可以自由的定義listview每一列的佈局,但當listview有大量的數據需要加載的時候,會佔據大量內存,影響性能,這時候就需要按需填充並重新使用view來減少對象的創建。

ListView加載數據都是在public View getView(int position, View convertView, ViewGroup parent) {}方法中進行的(要自定義listview都需要重寫listadapter:如BaseAdapter,SimpleAdapter,CursorAdapter的等的getvView方法),優化listview的加載速度就要讓convertView匹配列表類型,並最大程度上的重新使用convertView。

getview的加載方法一般有以下三種種方式:

最慢的加載方式是每一次都重新定義一個View載入佈局,再加載數據

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

View item = mInflater.inflate(R.layout.list_item_icon_text, null);

((TextView) item.findViewById(R.id.text)).setText(DATA[position]);

((ImageView) item.findViewById(R.id.icon)).setImageBitmap(

(position & 1) == 1 ? mIcon1 : mIcon2);

return item;

}



正確的加載方式是當convertView不爲空的時候直接重新使用convertView從而減少了很多不必要的View的創建,然後加載數據

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

if (convertView == null) {

convertView = mInflater.inflate(R.layout.item, parent, false);

}

((TextView) convertView.findViewById(R.id.text)).setText(DATA[position]);

((ImageView) convertView.findViewById(R.id.icon)).setImageBitmap(

(position & 1) == 1 ? mIcon1 : mIcon2);

return convertView;

}



最快的方式是定義一個ViewHolder,將convetView的tag設置爲ViewHolder,不爲空時重新使用即可

static class ViewHolder {

TextView text;

ImageView icon;

}



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

ViewHolder holder;



if (convertView == null) {

convertView = mInflater.inflate(R.layout.list_item_icon_text,

parent, false);

holder = new ViewHolder();

holder.text = (TextView) convertView.findViewById(R.id.text);

holder.icon = (ImageView) convertView.findViewById(R.id.icon);

convertView.setTag(holder);

} else {

holder = (ViewHolder) convertView.getTag();

}

holder.text.setText(DATA[position]);

holder.icon.setImageBitmap((position & 1) == 1 ? mIcon1 : mIcon2);

return convertView;

}



三種方式加載效率對比如下圖所示:

說明:上述三個例子代碼摘自google 2010 I/O大會



當處理一些耗時的資源加載的時候需要做到以下幾點,以使你的加載更快更平滑:

1.   適配器在界面主線程中進行修改

2.   可以在任何地方獲取數據但應該在另外一個地方請求數據

3.   在主界面的線程中提交適配器的變化並調用notifyDataSetChanged()方法
轉自:
http://blog.csdn.net/jacman/article/details/7087995
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章