博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
【Android】ListView 优化
阅读量:6587 次
发布时间:2019-06-24

本文共 2631 字,大约阅读时间需要 8 分钟。

重用 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);

转载地址:http://echno.baihongyu.com/

你可能感兴趣的文章
java8-谓词(predicate)
查看>>
简述vue-router实现原理
查看>>
用python模拟《流浪地球》的木星引力弹弓效应
查看>>
building xxx gradle project info的解决办法
查看>>
Netty+SpringBoot+FastDFS+Html5实现聊天App详解(四)
查看>>
【Leetcode】98. 验证二叉搜索树
查看>>
区块链共识问题都有什么?
查看>>
分布式事务中间件 Fescar - 全局写排它锁解读
查看>>
Vagrant (一) - 基本知识
查看>>
CSS选择器
查看>>
在 CentOS 7 上搭建 Jenkins + Maven + Git 持续集成环境
查看>>
一星期的学习
查看>>
Javascript 闭包详解
查看>>
数据结构与算法 | Leetcode 19. Remove Nth Node From End of List
查看>>
一起来读you don't know javascript(一)
查看>>
[LeetCode] 862. Shortest Subarray with Sum at Least K
查看>>
BIO、伪异步 IO、AIO和NIO
查看>>
【分享】终端命令工具 自动生成vue组件文件以及修改router.js
查看>>
吴恩达机器学习笔记-非监督学习
查看>>
《从0到1学习Flink》—— 如何自定义 Data Source ?
查看>>