[轉]listview加載性能優化ViewHolder

  當listview有大量的數據須要加載的時候,會佔據大量內存,影響性能,這時候就須要按需填充從新使用view來減小對象的建立佈局

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

getview的加載方法通常有如下三種種方式:優化

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

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

 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;get

}

 

正確的加載方式是當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()方法

相關文章
相關標籤/搜索