之所以说ListView这个控件很难用,就是因为它有很多细节可以优化,其中运行效率就是很重要的一点。目前我们ListView的运行效率是很低的,因为在FruitAdapter的getView()方法中,每次都将布局重新加载了一遍,当ListView快速滚动的时候,这就会成为性能的瓶颈。
仔细观察发现,getView()方法中还有一个convertView参数,这个参数用于将之前加载好的布局进行缓存,以便之后可以进行重用。修改FruitAdapter中的代码,如下所示:
package com.example.administrator.activitydemo;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ArrayAdapter;
import android.widget.ImageView;
import android.widget.TextView;
import java.util.List;
public class FruitAdapter extends ArrayAdapter<Fruit> {
private int resourceId;
private List<Fruit> list;
public FruitAdapter(Context context, int textViewResourceId, List<Fruit> list) {
super(context, textViewResourceId, list);
this.resourceId = textViewResourceId;
this.list = list;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
Fruit fruit = list.get(position);//获取当前项的Fruit实例
View view;
if (convertView==null){
view=LayoutInflater.from(getContext()).inflate(resourceId,parent,false);
}else {
view=convertView;
}
ImageView iv_name = (ImageView) view.findViewById(R.id.iv_name);//初始化图片
TextView tv_name = (TextView) view.findViewById(R.id.tv_name);//初始化文字
iv_name.setImageResource(fruit.getImageId());//为ImageView设置图片
tv_name.setText("" + fruit.getName());//为TextView设置文字
return view;
}
}
可以看到,现在我们在getView()方法中进行了判断,如果convertView为null,则使用LayoutInflater去加载布局,如果不为null,则直接对convertView进行重用。这样就大大提高了ListView的运行效率,在快速滚动的时候也可以表现出更好的性能。
不过,目前我们的这份代码还是可以继续优化的,虽然现在已经不会再重复去加载布局,但是每次在getView()方法中还是会调用View的findViewById()方法来获取一次控件的实例。我们可以借助一个ViewHolder来对这部分性能进行优化,修改FruitAdapter中的代码,如下所示:
package com.example.administrator.activitydemo;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ArrayAdapter;
import android.widget.ImageView;
import android.widget.TextView;
import java.util.List;
public class FruitAdapter extends ArrayAdapter<Fruit> {
private int resourceId;
private List<Fruit> list;
public FruitAdapter(Context context, int textViewResourceId, List<Fruit> list) {
super(context, textViewResourceId, list);
this.resourceId = textViewResourceId;
this.list = list;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
Fruit fruit = list.get(position);//获取当前项的Fruit实例
View view;
ViewHolder viewHolder;
if (convertView == null) {
view = LayoutInflater.from(getContext()).inflate(resourceId, parent, false);
viewHolder = new ViewHolder();
viewHolder.iv_name = (ImageView) view.findViewById(R.id.iv_name);//初始化图片
viewHolder.tv_name = (TextView) view.findViewById(R.id.tv_name);//初始化文字
view.setTag(viewHolder);//将ViewHolder存储在View中
} else {
view = convertView;
viewHolder = (ViewHolder) view.getTag();//重新获取ViewHolder
}
viewHolder.iv_name.setImageResource(fruit.getImageId());//为ImageView设置图片
viewHolder.tv_name.setText("" + fruit.getName());//为TextView设置文字
return view;
}
class ViewHolder {
ImageView iv_name;
TextView tv_name;
}
}
我们新增了一个内部类ViewHolder,用于对控件的实例进行缓存。当covertView为null的时候,创建一个ViewHolder对象,并将控件的实例放在ViewHolder里,然后调用View的setTag()方法,将ViewHolder重新取出。这样所有控件的实例都缓存在了ViewHolder里,就没必要每次都通过findViewById()方法来获取控件实例了。
通过这两步优化之后,我们ListView的运行效率就已经非常不错了。
效果图: