我们将问题定位在ListView的适配器中的getView(int position, View convertView, ViewGroup parent)中:
1. 最初的写法即加载子项布局View,获取View中的控件实例,对各控件实例进行赋值,最后再返回这个View。这样的写法每次都会重复加载布局,运行效率最低。
2. 仔细观察会发现,getView()方法中有一个convertView参数,这个参数用于将之前加载好的布局进行缓存,以便之后可以进行重用。优化如下:
...
View view;
if(convertView == null){
view = LayoutInflater.from(getContext()).inflate(resourceId, parent, false);
}else {
view = convertView;
}
...
3. 虽然现在不会再重复去加载布局,但是每次的getView()方法中还是会调用View的findViewById()方法来获取一次控件的实例。我们可以借助一个ViewHolder来对这部分性能进行优化。优化如下:
...
View view;
ViewHolder viewHolder;
if(convertView == null){
view = LayoutInflater.from(getContext()).inflate(resourceId, parent, false);
viewHolder = new ViewHolder();
viewHolder.imageView = (ImageView)view.findViewById(R.id.imageView);
viewHolder.textView = (TextView)view.findViewById(R.id.textView);
view.setTag(viewHolder);
}else {
view = convertView;
viewHolder = (ViewHolder)view.getTag();
}
viewHolder.imageView.set....
viewHolder.textView.set...
...
class ViewHolder{
ImageView imageView;
TextView textView;
}
我们新增了一个内部类ViewHolder,用于对控件的实例进行缓存。当convertView为null时,创建一个ViewHolder对象,并将控件的实例都存放在ViewHolder里,然后调用View的setTag()方法,将ViewHolder对象存储中View中。当convertView不为null的时候,则调用View的getTag()方法,把ViewHolder重新取出。这样所有的控件实例都缓存在了ViewHolder里,就没有必要每次都通过findViewById()方法来获取控件实例了。
通过这两步优化之后,ListView的运行效率就已经非常不错了。