Android中ViewHolder的使用

RungBy 2014-01-02

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。

相关推荐