ListView优化:

目前我们ListView 的运行效率是很低的,因为在FruitAdapter 的getView()方法中每次都将布局重新加载了一遍,当ListView 快速滚动的时候这就会成为性能的瓶颈。

仔细观察,getView()方法中还有一个convertView 参数,这个参数用于将之前加载好的布局进行缓存,以便之后可以进行重用。修改FruitAdapter 中的代码,如下所示:

 

@Override

    public View getView(int position, View convertView, ViewGroup parent) {

        Fruit fruit = getItem(position);

        View view;

        if( convertView == null ){

            view = LayoutInflater.from(getContext()).inflate(resourceId, null);

        }else {

            view = convertView;

        }

        ImageView imageView = (ImageView)view.findViewById(R.id.fruit_image);

        TextView textView = (TextView)view.findViewById(R.id.fruit_name);

        imageView.setImageResource(fruit.getImageId());

        textView.setText(fruit.getName());

        return view;

    }

 

 

可以看到,现在我们在getView()方法中进行了判断,如果convertView 为空,则使用LayoutInflater 去加载布局,如果不为空则直接对convertView 进行重用。这样就大大提高了ListView 的运行效率,在快速滚动的时候也可以表现出更好的性能。

不过,目前我们的这份代码还是可以继续优化的,虽然现在已经不会再重复去加载布局,但是每次在getView()方法中还是会调用View 的findViewById()方法来获取一次控件的实例。我们可以借助一个ViewHolder 来对这部分性能进行优化,修改FruitAdapter 中的代码,如下所示:

 

public class FruitAdapter extends ArrayAdapter<Fruit> {

 

    private int resourceId;

 

    public FruitAdapter(Context context, int resource, List<Fruit> objects) {

        super(context, resource, objects);

        this.resourceId = resource;

    }

 

    @NonNull

    @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.imageView = (ImageView)view.findViewById(R.id.fruit_image);

            viewHolder.textView = (TextView)view.findViewById(R.id.fruit_name);

            view.setTag(viewHolder); // 将ViewHolder存储在View中

        }else {

            view = convertView;

            viewHolder = (ViewHolder)view.getTag();// 重新获取ViewHolder

        }

 

        viewHolder.imageView.setImageResource(fruit.getImageId());

        viewHolder.textView.setText(fruit.getName());

        return view;

    }

 

    class ViewHolder{

        ImageView imageView;

        TextView textView;

    }

}

 

我们新增了一个内部类ViewHolder,用于对控件的实例进行缓存。当convertView为空的时候,创建一个ViewHolder 对象,并将控件的实例都存放在ViewHolder 里,然后调用View 的setTag()方法,将ViewHolder 对象存储在View 中。当convertView 不为空的时候则调用View 的getTag()方法,把ViewHolder 重新取出。这样所有控件的实例都缓存在了ViewHolder 里,就没有必要每次都通过findViewById()方法来获取控件实例了。

posted @ 2017-04-17 11:57  rorshach  阅读(208)  评论(0编辑  收藏  举报