zoukankan      html  css  js  c++  java
  • 【Android】ListView 优化

    重用 ListView Item

    ListView创建时其会创建屏幕可容纳数量的 Item。ListView 滚动时,刚消失的 item 会被保存到回收池中。新出现的 item 从回收池中获取避免反复创建,这个回收池由 ListView 维护。

    从回收池取出 item 会传递给 Adapter 的 getView() 方法的第二个參数。假设回收池中没有内容就传递一个 null。所以在 getView() 方法中假设第二个參数不为 null,就重用传入的 ListView Item。这能够极大的提高 ListView 性能。

    public void getView(int position, View convertView, ViewGroup parent) {
        View view = convertView;
        if (view == null) {
             // 创建新的 ListView item
             view = LayoutInflator.from(mContext).inflate(...);
         }
    
        /* 更新 ListView Item */
        TextView textView = (TextView)view.findViewById(R.id.textview);
        textView.setText("new data");
    
        return view;
    }

    使用ViewHold中避免每次调用findViewById()

    使用上面重用 ListView Item 的方法已经能够大幅提高 ListView 的效率了,但还存在一些能够改进的地方,上面每次更新 ListView Item 数据时,都要通过 View 的 findViewById() 方法定位每一个子控件, findViewById() 会沿着ListView Item 的控件布局结构遍历每一个控件直到找到指定 id 的控件,这是比較耗时的,尤其是布局比較复杂时。

    优化方法非常easy,在每次创建新的 ListView Item 时保存通过 findViewById() 找到的每一个子控件的引用。这些控件引用能够保存在一个单独的对象中。一般命名为 ViewHolder,然后将 ViewHolder 对象存储在 ListView item 中(通过 View 的 setTag() 方法。该方法能够在 View 中存储额外数据)。下次能够直接从 ListView item 中取得。

    public void getView(int position, View convertView, ViewGroup parent) {
        View view = convertView;
        if (view == null) {
             // 创建新的 ListView item
             view = LayoutInflator.from(mContext).inflate(...);
             // 保存控件引用在 ViewHolder 对象中
             ViewHolder holder = new ViewHolder();
             holder.textView = (TextView)view.findViewById(R.id.textview);
             // 将 ViewHolder 对象保存在 ListView Item 中
             view.setTag(holder);
         }
    
        /* 更新 ListView Item */
        ViewHolder holder = (ViewHolder)view.getTag();
        holder.textView.setText("new data");
    
        return view;
    }
    
    private static class ViewHolder {
        public TextView textView;
    }

    固定ListView Item的大小

    因为 ListView 高度或宽度不固定(布局參数中的宽或高指定为 wrap_content),导致 ListView 重绘时须要又一次计算 ListView Item 的大小,从而导致 getView() 方法反复调用(这里是指针对同一个ListView item)。这对 ListView 的性能有显著的影响,解决的方法就是指定 ListView 布局參数中的高和宽为固定大小或 match_parent

    避免更新同样数据

    当 ListView Item 中包括图片控件时, 在 getView() 中更新数据通常是依据图片的 URI 取得 Bitmap 然后设置图片。假设 ImageView 显示的图片地址和要更新的图片地址同样,就全然没有必要做更新操作。

    能够採用和 ViewHolder 同样的方式。将图片的 URI 地址保存在 ImageView 的 tag 中,每次更新 ImageView 时先推断图片地址是否发生变化,仅仅有发生变化时採取获取新的 Bitmap 更新。

    String newImgUrl = ...;
    String imgUrl = (String)imageView.getTag();
    if (imageUrl == null || !imageUrl.equal(newImgUrl)) {
        // 请求 newImgUrl 相应的图片
        Bitmap bitmap = ...;
        imageView.setImageBitmap(bitmap);
        imageView.setTag(newImgUrl);
    }

    限制 ListView 的滚动速度

    ListView 默认的滚动速度是比較快的,假设 ListView 滚动速度慢一点,那么每一个 ListView Item 就有很多其它的载入时间。这也能够使 ListView 看上去更加流畅。

    以下代码将 ListView 的滚动速度减慢为原来的 1/10:

    listview.setFriction(ViewConfiguration.getScrollFriction() * 10);
  • 相关阅读:
    Broadcom 43228 Kali Linux Ubuntu
    linux 栈空间查看和修改
    mininet 操作命令
    linux shell note
    进程与线程的区别
    JAVA基础--JAVA 集合框架(泛型、file类)
    HashMap的实现原理
    Java 流(Stream)、文件(File)和IO
    总结接口和抽象类的异同
    Java 泛型
  • 原文地址:https://www.cnblogs.com/yutingliuyl/p/7111616.html
Copyright © 2011-2022 走看看