Android中GridView和ListView是最重要的两种显示内容的控件,在Android中,受限于手机屏幕的大小,使的不能像电脑一样同时显示多项内容。
这两者都需要一个adapter,可以是自定义适配器,也可以是BaseAdapter,SimpleAdapter,CursorAdapter。自定义适配器中,最为重要的是getView()方法,在该方法中有一个convertView参数,该参数就是用来加载数据时的View。
在ListView中有时我们需要加载大量数据,如果每次都创建一个View,这样会占据大量内存,影响性能。这时就可以考虑用ViewHolder了。
ViewHolder不是Android的开发API,而是一种设计方法,就是设计个静态类,缓存一下,省得Listview更新的时候,还要重新操作。
下面以listView加载方式比较一下viewholder的便利。
viewholder的方式:将convetView的tag设置为ViewHolder,不为空时重新使用即可:
class ViewHolder{
ImageView img;
TextView sitekey;
TextView partname;
TextView price;
TextView value;
TextView quantity_sold;
TextView end_date;
TextView jiantou;
}
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder = new ViewHolder();
if(convertView==null){
convertView = inflater.inflate(R.layout.good_list_item, null, false);
holder.img = (ImageView) convertView.findViewById(R.id.img);
holder.sitekey = (TextView) convertView.findViewById(R.id.sitekey);
holder.partname = (TextView) convertView.findViewById(R.id.partname);
holder.price = (TextView) convertView.findViewById(R.id.price);
holder.value = (TextView) convertView.findViewById(R.id.value);
holder.quantity_sold = (TextView) convertView.findViewById(R.id.quantity_sold);
holder.end_date = (TextView) convertView.findViewById(R.id.end_date);
holder.jiantou = (TextView) convertView.findViewById(R.id.jiantou);
convertView.setTag(holder);
}else{
holder = (ViewHolder) convertView.getTag();
}
//设置holder
holder.img.setImageResource(R.drawable.ic_launcher);
holder.sitekey.setText(list.get(position).sitekey);
holder.partname.setText(list.get(position).partname);
holder.price.setText("$"+list.get(position).price);
holder.value.setText("$"+list.get(position).value);
holder.quantity_sold.setText("已售出:"+list.get(position).quantity_sold);
holder.end_date.setText("截止日期:"+list.get(position).end_date);
return convertView;
}
错误的加载方式:每一次都重新定义一个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;
}
convertView 在API中的解释是The old view to reuse, if possible, 第一次getView时还没有convertView,这时你便创建了一个新的view,下次getView时就有这个“旧的”convertView了 setTag的作用才是把查找的view通过ViewHolder封装好缓存起来方便多次重用,当需要时可以getTag拿出来
当你的listview里布局多样化的时候 viewholder的作用就有比较明显的体现了。 当然了,单一模式的布局一样有性能优化的作用 只是不直观。 假如你2种模式的布局 当发生回收的时候 你会用setTag分别记录是哪两种 这两种模式会被封装到viewholder中进行保存方便你下次使用。 所以当加载大量数据时,最好使用ViewHolder。