使得ListView滑动顺畅
最后更新于:2022-04-01 01:46:49
> 编写:[allenlsy](https://github.com/allenlsy) - 原文:[http://developer.android.com/training/improving-layouts/smooth-scrolling.html](http://developer.android.com/training/improving-layouts/smooth-scrolling.html)
保持程序流畅的关键,是让主线程(UI 线程)不要进行大量运算。你要确保在其他线程执行磁盘读写、网络读写或是 SQL 操作等。为了测试你的应用的状态,你可以启用 [StrictMode](http://developer.android.com/reference/android/os/StrictMode.html)。
### 使用后台线程
你应该把主线程中的耗时间的操作,提取到一个后台线程(也叫做“工人线程”,英文 worker thread)中,使得主线程只关注 UI 绘画。很多时候,使用 [AsyncTask](http://developer.android.com/reference/android/os/AsyncTask.html) 是一个简单的在主线程以外进行操作的方法。系统使用一个全局队列来排列所有的 AsyncTask,并且一次运行其中的 [execute()](http://developer.android.com/reference/android/os/AsyncTask.html#execute(Params...)) 方法。这个行为是全局的,这意味着你不需要考虑自己定义线程池的事情。
在以下的例子中,一个 AsyncTask 被用于在后台线程载入图片,并在载入完成后把图片放入 UI 。当图片正在载入时,它还回显示一个进度转盘。
~~~
// Using an AsyncTask to load the slow images in a background thread
new AsyncTask<ViewHolder, Void, Bitmap>() {
private ViewHolder v;
@Override
protected Bitmap doInBackground(ViewHolder... params) {
v = params[0];
return mFakeImageLoader.getImage();
}
@Override
protected void onPostExecute(Bitmap result) {
super.onPostExecute(result);
if (v.position == position) {
// If this item hasn't been recycled already, hide the
// progress and set and show the image
v.progress.setVisibility(View.GONE);
v.icon.setVisibility(View.VISIBLE);
v.icon.setImageBitmap(result);
}
}
}.execute(holder);
~~~
从 Android 3.0 (API level 11) 开始, AsyncTask 有个新特性,那就是它可以在多个 CPU 核上运行。你可以调用 [`executeOnExecutor()`](http://developer.android.com/reference/android/os/AsyncTask.html#executeOnExecutor(java.util.concurrent.Executor,%20Params...) 来自动根据核数,在多核上执行任务,而不是调用 [`execute()`](http://developer.android.com/reference/android/os/AsyncTask.html#execute(Params...)。
### 在 View Holder 中填入视图对象
你的代码可能在 ListView 滑动时经常使用 `findViewById()`,这样会降低性能。即使是 Adapter 返回一个用于回收的 inflate 后的视图,你仍然需要查看这个元素并更新它。避免频繁调用 `findViewById()` 的方法之一,就是使用 View Holder(视图占位符)设计模式。
一个 ViewHolder 对象存储了他的标签下的每个视图。这样你不用频繁查找这个元素。第一,你需要创建一个类来存储你会用到的视图。比如:
~~~
static class ViewHolder {
TextView text;
TextView timestamp;
ImageView icon;
ProgressBar progress;
int position;
}
~~~
然后,在 Layout 的类中生成一个 ViewHolder 对象:
~~~
ViewHolder holder = new ViewHolder();
holder.icon = (ImageView) convertView.findViewById(R.id.listitem_image);
holder.text = (TextView) convertView.findViewById(R.id.listitem_text);
holder.timestamp = (TextView) convertView.findViewById(R.id.listitem_timestamp);
holder.progress = (ProgressBar) convertView.findViewById(R.id.progress_spinner);
convertView.setTag(holder);
~~~
这样你就可以轻松获取每个视图,而不是用 `findViewById()` 来不断查找视图,节省了宝贵的运算时间。