之前在详解中对于ListView的用法效率并不是最高的,因为在FruitAdapter的getView()方法中,每次当子项滑动到屏幕内时,都需要重新加载子项布局文件。同样,每当子项滑动到屏幕当中,每次都需要View的findViewById()方法来获取控件实例。
- 解决每次均需要加载子项布局文件问题的方法
public View getView(int position,View convertView,ViewGroup parent)
在getView()方法中,第二个参数convertView,这个参数就是用来对已经加载过的布局进行缓存,这样当缓存的子项再次滑动到屏幕被时,就不用再重新加载布局文件了。
if (convertView == null){
view = LayoutInflater.from(getContext()).inflate(resourceId, null);
}else {
view = convertView;
}
首先判断convertView是否为空,即判断是否已经进行过缓存,当子项布局文件没有进行缓存时,才将加载此子项布局文件。
2.解决每次均需要获取控件实例问题的方法
与上个问题同理,首先想到的是将控件实例进行缓存,即可以提高效率。
我们新建一个内部类ViewHolder,用于对控件的实例进行缓存。当convertView为空时,就将控件实例存放于ViewHolder中,调用View的setTag()方法将Viewholder存于View中。当convertView不为空时,则调用View的getTag()方法将ViewHolder重新取出,这样所有的控件实例都存放于Viewholder中。这样当屏幕不断滑动时,就没有必要不断的通过findViewById()重新获取控件实例。
@Override
public View getView(int position,View convertView,ViewGroup parent){
Fruit fruit = getItem(position);//获取当前项的实例
View view;
ViewHolder viewHolder;
if (convertView == null){
view = LayoutInflater.from(getContext()).inflate(resourceId, null);
viewHolder = new ViewHolder();
viewHolder.fruitImage = (ImageView) view.findViewById(R.id.fruit_image);
viewHolder.fruitName = (TextView) view.findViewById(R.id.fruit_name);
view.setTag(viewHolder);
}else {
view = convertView;
viewHolder = (ViewHolder) view.getTag();//重新获取ViewHolder
}
viewHolder.fruitImage.setImageResource(fruit.getImageId());
viewHolder.fruitName.setText(fruit.getName());
return view;
}
class ViewHolder{
ImageView fruitImage;
TextView fruitName;
}
通过这两次优化,ListView的运行效率提高了不少。尤其是在快速滑动屏幕时,效率差距更为明显。