Listview的優化

在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()方法

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章