平滑滚动的关键ListView是保持应用程序的主线程(UI线程)免于繁重的处理。确保您在单独的线程中执行任何磁盘访问,网络访问或SQL访问。要测试您的应用的状态,您可以启用StrictMode。
使用后台线程
使用后台线程(“工作线程”)可以消除主线程的负担,因此可以专注于绘制UI。在许多情况下,使用AsyncTask提供了一种简单的方式在主线程之外执行您的工作。AsyncTask自动排队所有execute()请求并连续执行。这种行为对于特定的进程是全局性的,并且意味着您不必担心创建自己的线程池。
在下面的示例代码中,an 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级别11)开始,可以使用额外的功能,AsyncTask因此您可以使其能够跨多个处理器内核运行。取而代之的是execute()你可以指定executeOnExecutor()多个请求,并且可以同时执行,具体取决于可用的内核数量。
在视图持有者中保持视图对象
您的代码可能会findViewById()在滚动过程中频繁地调用ListView,这会降低性能。即使Adapter返回的是回收的膨胀视图,您仍然需要查找元素并进行更新。重复使用的一种方法findViewById()是使用“视图持有者”设计模式。
一个ViewHolder对象将每个组件视图存储在Layout的标签字段中,因此您可以立即访问它们而无需重复查看它们。首先,您需要创建一个类来保存一组精确的视图。例如:
static class ViewHolder {
TextView text;
TextView timestamp;
ImageView icon;
ProgressBar progress;
int position;
}
然后填充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);
现在,您可以轻松访问每个视图而无需查找,从而节省宝贵的处理器周期。
有关与此主题相关的更多信息,请参阅 我的列表为什么是黑色的?Android优化。
联系我
QQ:94297366
微信打赏:https://pan.baidu.com/s/1dSBXk3eFZu3mAMkw3xu9KQ扫描二维码关注公众号,回复: 1775622 查看本文章
公众号推荐: