• 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()方法来获取控件实例了。

  • 相关阅读:
    大学生创业不可或缺的六项品质
    C#的9*9乘法表!
    湖北武汉的进来!看看!
    每束焰火都装了电脑芯片
    学习C#之旅 魔泡排序
    学习C#之旅(C#语言基础,运算符)
    主攻ASP.NET.3.5.MVC架构之重生: URL Routing (三)
    主攻ASP.NET.3.5.MVC架构之重生: LINQ(六)
    【HDU】3341 Lost's revenge
    【HDU】2243 考研路茫茫――单词情结
  • 原文地址:https://www.cnblogs.com/itfenqing/p/6722237.html
Copyright © 2020-2023  润新知