在android開發(fā)中Listview是一個很重要的組件,它以列表的形式根據(jù)數(shù)據(jù)的長自適應展示具體內(nèi)容,用戶可以自由的定義listview每一列的布局,但當listview有大量的數(shù)據(jù)需要加載的時候,會占據(jù)大量內(nèi)存,影響性能,這時候就需要按需填充并重新使用view來減少對象的創(chuàng)建。
ListView加載數(shù)據(jù)都是在public View getView(int position, View convertView, ViewGroup parent) {}方法中進行的(要自定義listview都需要重寫listadapter:如BaseAdapter,SimpleAdapter,CursorAdapter的等的getvView方法),優(yōu)化listview的加載速度就要讓convertView匹配列表類型,并最大程度上的重新使用convertView。
getview的加載方法一般有以下三種種方式:
最慢的加載方式是每一次都重新定義一個View載入布局,再加載數(shù)據(jù)
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的創(chuàng)建,然后加載數(shù)據(jù)
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. 可以在任何地方獲取數(shù)據(jù)但應該在另外一個地方請求數(shù)據(jù)
3. 在主界面的線程中提交適配器的變化并調(diào)用notifyDataSetChanged()方
法