Android中ViewHolder的使用
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。