`
zhouxiaoli521
  • 浏览: 561893 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

【译】利用多线程提高程序性能(for Android)

阅读更多

[原作者是Gilles Debunne,身份不详...]
要想搞出一个反应迅速的Android应用程序,一个很好的做法就是确保在主UI线程里执行尽量少的代码。任何有可能花费较长时间来执行的代码如果在主UI线程执行,则会让程序挂起无法响应用户的操作,所以应该放到一个单独的线程里执行。典型的例子就是与网络通信相关的操作了,因为通过网络收发信息的快慢我们无法预测,有可能“biu”地一下就搞定了,也有可能磨磨唧唧半天。用户心情好的话可能会容忍一点点迟延,而且前提是你给出了必要的提示,但是一个看上去根本不动貌似嗝儿屁的程序……(译注:就好比Ajax技术出现之前的网页,用户可以习惯短时间的载入,但是一个载入了半天都是空白的浏览器窗口就常常让那个拨号时代的我们感到困惑和抓狂。)
在这篇文章中,我们将创建一个简单的图片下载程序来演示一下多线程模式。我们将从网上下载一坨图片,然后用这些图片生成一个缩略图列表。创建一个异步工作的任务,让它在后台下载图片,会让我们的程序看上去更快。(译注:这里我加上“看上去”,因为我认为所谓多线程让程序更快,更多的意义在于“提高对用户操作的响应”。包括本文题目,所谓的“高性能”,主要指的还是避免UI的硬直(格斗游戏术语,请自行google)、挂起。毕竟多线程无法避免代码固有的主要资源开销。)

一个图片下载器

从web下载图片很简单,使用SDK提供的HTTP相关的类即可实现。下面是一个简单的实现。
(译注:下面用到的AndroidHttpClient等类从2.2版,也就是API Level 8才开始提供。请2.1以下各位从代码领会精神即可。直接用HttpClient应该亦可实现。)

static Bitmap downloadBitmap(String url) {
final AndroidHttpClient client = AndroidHttpClient.newInstance("Android");
final HttpGet getRequest = new HttpGet(url);

首先我们创建了一个HTTP客户端和HTTP请求。如果请求成功,就把响应中包含的图片内容解码成位图格式并返回,以备后续使用。另外补充一句,为了让程序可以访问网络,必须在程序的manifest文件中声明使用INTERNET。
注意:旧版的BitmapFactory.decodeStream有个bug,可能使得在网络较慢的时候无法正常工作。可以使用FlushedInputStream(inputStream)代替原始的inputStream来解决这个问题。下面是这个helper class的实现:

static class FlushedInputStream extends FilterInputStream {
public FlushedInputStream(InputStream inputStream) {
super(inputStream);
}

这个类可以保证skip()确实跳过了参数提供的字节数,直到流文件的末尾。

try {
HttpResponse response = client.execute(getRequest);
final int statusCode = response.getStatusLine().getStatusCode();
if (statusCode != HttpStatus.SC_OK) {
Log.w("ImageDownloader", "Error " + statusCode + " while retrieving bitmap from " + url);
return null;
}

final HttpEntity entity = response.getEntity();
if (entity != null) {
InputStream inputStream = null;
try {
inputStream = entity.getContent();
final Bitmap bitmap = BitmapFactory.decodeStream(inputStream);
return bitmap;
} finally {
if (inputStream != null) {
inputStream.close();
}
entity.consumeContent();
}
}
} catch (Exception e) {
// Could provide a more explicit error message for IOException or IllegalStateException
getRequest.abort();
Log.w("ImageDownloader", "Error while retrieving bitmap from " + url, e.toString());
} finally {
if (client != null) {
client.close();
}
}
return null;
}

@Override
public long skip(long n) throws IOException {
long totalBytesSkipped = 0L;
while (totalBytesSkipped < n) {
long bytesSkipped = in.skip(n - totalBytesSkipped);
if (bytesSkipped == 0L) {
int byte = read();
if (byte < 0) {
break; // we reached EOF
} else {
bytesSkipped = 1; // we read one byte
}
}
totalBytesSkipped += bytesSkipped;
}
return totalBytesSkipped;
}
}

如果你在ListAdapter的getView方法中直接使用上面的downloadBitmap方法,结果可以想象的出,随着我们滚动屏幕,一定是一顿一顿很不爽的。因为每显示一个新的view,都必须等待一张图片完成下载,势必会影响滚屏的流畅度。

正是因为这想都想得出来的糟糕体验,AndroidHttpClient根本就不允许在主线程里启动!上面的代码在主线程里将会提示“本线程无法进行HTTP请求”。如果你不见棺材不落泪,说啥也要亲手试试这糟糕的用户体验的话,可以用DefaultHttpClient代替AndroidHttpClient,给自己一个交代。

异步任务

AsyncTask类提供了一个从主线程生成新任务的方法。让我们创建一个ImageDownloader类来负责生成任务。这个类将提供一个download方法,从指定URL下载图片,并在ImageView里显示出来。

public class ImageDownloader {

BitmapDownloaderTask继承自AsyncTask。它真正执行图片下载的任务。任务通过execute方法启动,该方法是立即返回的,从而使得调用它的主线程代码可以迅速执行完毕。这正是我们使用AsyncTask的意义所在。下面是BitmapDownloaderTask的实现:

class BitmapDownloaderTask extends AsyncTask {
private String url;
private final WeakReference imageViewReference;

doInBackground方法是真正在单独进程中执行异步任务的代码。它调用前面介绍的downloadBitmap方法,完成下载,取得位图。
onPostExecute在任务结束后由主线程调用。它通过传入的参数得到下载回来的位图,并设置到ImageView显示(该ImageView在实例化BitmapDownloaderTask时传入)。需要注意的是这里对ImageView的引用是以WeakReference的形式保存在BitmapDownloaderTask实例里,所以在下载过程中如果activity被关掉,无法阻止activity里的ImageView被回收。因此我们必须在使用前检查imageViewReference和imageview是否为空。
这个简单的小例子演示了如何使用AsyncTask。如果你亲自动手实验一下,应该会发现这短短几行代码显著地改善了ListView的滚屏体验。推荐阅读developer.android.com的文章《
Painless threading》来学习AsyncTasks的更多细节。
但是,这个基于ListView的例子暴露出一个问题。出于对内存的利用效率考虑,ListView会在用户滚屏的时候对view进行循环再利用。如果用户快速猛烈发飙般地滚屏,一个ImageView对象将会被反复使用多次。每一次它被显示出来,都会触发生成一个下载图片的任务,从而改变这个ImageView的显示内容。那么问题在哪呢?跟大部分并行程序一样,关键问题在于顺序。在我们这个例子中,没有采取任何措施保证所有下载任务按顺序完成,换句话说,无法保证先启动的任务先完成,后启动的任务后完成。这样就导致显示在list中的图片可能来自之前的任务,该任务因为花费的时间更长,所以最后结束,最终导致预期外的结果。如果你要下载的图片们是一次性绑定到一坨ImageView的,那么就不存在问题,但我们还是从大局出发,为了通用的情况,修正一下吧。

public void download(String url, ImageView imageView) {
BitmapDownloaderTask task = new BitmapDownloaderTask(imageView);
task.execute(url);
}
}

/* class BitmapDownloaderTask, see below */
}

public BitmapDownloaderTask(ImageView imageView) {
imageViewReference = new WeakReference(imageView);
}

@Override
// Actual download method, run in the task thread
protected Bitmap doInBackground(String... params) {
// params comes from the execute() call: params[0] is the url.
return downloadBitmap(params[0]);
}

@Override
// Once the image is downloaded, associates it to the imageView
protected void onPostExecute(Bitmap bitmap) {
if (isCancelled()) {
bitmap = null;
}

if (imageViewReference != null) {
ImageView imageView = imageViewReference.get();
if (imageView != null) {
imageView.setImageBitmap(bitmap);
}
}
}
}

并发处理

要想解决上面提到的问题,我们需要知道并保存下载任务的顺序,以保证最后启动的任务最后结束,并完成对ImageView的更新。要达到这个目的,让每个ImageView记住自己的最后一个下载任务就可以了。我们使用一个专用的Drawable类给ImageView添加这份信息。这个Drawable类将在下载过程中临时绑定到ImageView。下面是这个DownloadedDrawable类的代码:

static class DownloadedDrawable extends ColorDrawable {
private final WeakReference bitmapDownloaderTaskReference;

这个实现方法引入了一个ColorDrawable,这会导致ImageView在下载过程中显示黑色的背景。需要的话,可以使用一个显示“下载中…”之类的图片代替之,换取更友好的用户界面。再提一遍,注意使用WeakReference来降低与对象实例的耦合。
让我们修改之前的代码来让这个类起作用。首先,download方法将创建这个类的实例并绑定到ImageView:

public void download(String url, ImageView imageView) {
if (cancelPotentialDownload(url, imageView)) {
BitmapDownloaderTask task = new BitmapDownloaderTask(imageView);
DownloadedDrawable downloadedDrawable = new DownloadedDrawable(task);
imageView.setImageDrawable(downloadedDrawable);
task.execute(url, cookie);
}
}

cancelPotentialDownload方法将在一个新的下载开始前取消尚在进行中的下载任务。注意,这并不足以保证新开始的下载任务得到的图片一定能够被显示,因为之前的任务可能已经完成了,处于等待onPostExecute方法执行的时间点,而这个onPostExecute方法还是有可能在新任务的onPostExecute方法之后执行。

private static boolean cancelPotentialDownload(String url, ImageView imageView) {
BitmapDownloaderTask bitmapDownloaderTask = getBitmapDownloaderTask(imageView);

cancelPotentialDownload调用AsyncTask类的cancel方法来停止进行中的下载任务。大部分情况下它返回true,所以调用它的download方法中可以开始新的下载。唯一的例外情况是如果进行中的下载任务与新任务请求的是同一个URL,我们就不取消旧任务了,让它继续下载。注意在我们这个实现方法中,如果ImageView被回收了,与其关联的下载不会停止(可以借助RecyclerListener实现)。
这个方法还调用了一个helper函数getBitmapDownloaderTask。代码很直观,不做赘述:

private static BitmapDownloaderTask getBitmapDownloaderTask(ImageView imageView) {
if (imageView != null) {
Drawable drawable = imageView.getDrawable();
if (drawable instanceof DownloadedDrawable) {
DownloadedDrawable downloadedDrawable = (DownloadedDrawable)drawable;
return downloadedDrawable.getBitmapDownloaderTask();
}
}
return null;
}

最后,必须修改一下onPostExecute方法,保证只在ImageView尚与下载进程关联的情况下绑定位图到ImageView:

if (imageViewReference != null) {
ImageView imageView = imageViewReference.get();
BitmapDownloaderTask bitmapDownloaderTask = getBitmapDownloaderTask(imageView);
// Change bitmap only if this process is still associated with it
if (this == bitmapDownloaderTask) {
imageView.setImageBitmap(bitmap);
}
}

嗯,做了这些修改之后,我们的ImageDownloader类基本可以提供预期的服务了。你可以在自己的项目中灵活运用这些代码或者它演示的异步思想,改善用户体验。

public DownloadedDrawable(BitmapDownloaderTask bitmapDownloaderTask) {
super(Color.BLACK);
bitmapDownloaderTaskReference =
new WeakReference(bitmapDownloaderTask);
}

public BitmapDownloaderTask getBitmapDownloaderTask() {
return bitmapDownloaderTaskReference.get();
}
}

if (bitmapDownloaderTask != null) {
String bitmapUrl = bitmapDownloaderTask.url;
if ((bitmapUrl == null) || (!bitmapUrl.equals(url))) {
bitmapDownloaderTask.cancel(true);
} else {
// The same URL is already being downloaded.
return false;
}
}
return true;
}

Demo

本文的源代码可以从Google Code获取。你可以在本文提到的三种实现方式(非异步、无并发处理以及最终版本)中切换、比较。注意,缓存大小已经被限制到10张图片以便更好地演示可能出现的问题。

进一步的工作

文中代码为了集中讨论并行问题而做了简化,因此缺少很多功能。首先ImageDownloader类应该利用缓存,特别是与ListView结合使用的时候。因为ListView在用户上下往返滚屏的时候会多次显示相同图片,而缓存可以大大降低开销。通过使用一个基于LinkedHashMap(该hashmap提供从URL到Bitmap SoftReference的映射)的LRU缓存可以很容易地实现这一点。更加复杂的缓存机制还可以依赖于本地存储。缩略图的创建、图片缩放等功能也可以考虑加进来。
本文代码已经考虑到了下载错误和超时的情况。这些情况下将会返回一个空位图。你也可以显示一张带有提示信息的图片。
本文示例的HTTP请求很简单。根据实际情况的不同(大都依赖于服务器端),可以在HTTP请求中加入各种参数或者cookie等等。
本文使用的AsyncTask类是一个把任务从主线程分离出来很简单方便的途径。你可能会用到Handler类来实现对任务流程更好的控制,比如控制并行的下载线程数,等等。

原文地址:猛击这里(在墙的那边)

 

http://9601.org/join/?p=174

分享到:
评论

相关推荐

    VB for Android12.12

    5. **多线程编程**:VB在Android平台上同样需要处理多线程问题,理解如何在Android环境下创建并管理线程,确保应用的高效运行。 6. **调试与测试**:学习使用VB for Android提供的调试工具,进行应用的错误检查和...

    android 单线程 多线程下载

    在Android开发中,数据的加载和...总之,理解和掌握Android中的单线程和多线程下载,不仅能够提高应用程序的性能,还能提升用户的体验。在实践中,我们需要不断探索和优化,利用各种工具和库来简化和增强我们的代码。

    VLC for android 2014.9

    2. **多线程编程**:播放器通常涉及多个线程,如解码线程、渲染线程等,理解如何协调它们以保证流畅播放。 3. **文件I/O**:了解VLC如何读取和处理本地及网络媒体文件。 4. **UI设计**:分析VLC的用户界面设计,...

    MinaServer for Android

    在Android上运行服务端程序面临一些挑战,如资源限制、多线程管理以及后台服务的生命周期管理。MinaServer通过优化内存管理和线程模型,能在各种Android设备上稳定运行,包括三星平板、小米手机、华为手机、三星手机...

    mono for android demo 50个

    8. **多线程**:学习在 Android 上使用线程和线程池,以及如何处理线程同步问题。 9. **网络通信**:使用 HttpClient 或其他库进行网络请求,实现数据的获取和发送。 10. **SQLite 数据库**:了解如何在 Android ...

    lockscreen for android 2.2

    此外,为了保证兼容性和性能,开发者还需要关注Android的多线程编程、内存管理、权限控制等方面,确保应用能在不同设备和版本上稳定运行。 在具体到“LockScreenMedical”这个子文件来看,可能是一个医疗主题的锁屏...

    Mono for Android和MonoDroid学习笔记全套

    5. **多线程与异步编程**:Mono for Android支持.NET的Task-based Asynchronous Pattern (TAP),笔记会讲解如何在Android环境中进行异步操作,提高应用性能。 6. **网络通信**:了解如何使用HttpClient或WebClient...

    Learn Java for Android

    在本书的高级主题部分,Java的多线程编程会是重要的内容,因为Android应用通常需要在多线程环境中运行以提升性能和响应能力。此外,作者还会介绍Java的网络编程能力,这对于开发涉及数据传输的应用程序是必要的。...

    libjpeg-turbo for android

    此外,还可以通过多线程技术,进一步提高处理多张图片的并发能力。 6. **安全注意事项** 尽管libjpeg-turbo性能强大,但在实际应用中,应注意防止内存溢出和缓冲区溢出等安全问题。正确处理异常和错误情况,确保...

    安卓Android源码——ipcamera-for-android 手机变成IP Camera.zip

    7. **多线程和异步处理**:处理摄像头数据和网络通信通常涉及多线程,以避免阻塞UI主线程,因此需要理解Android的Handler、Looper和AsyncTask等机制。 8. **Socket编程**:如果项目使用自定义的网络协议,可能需要...

    Boost-for-Android-master.zip

    1. **性能提升**:Boost库中的许多组件都是高度优化的,比如多线程支持、内存管理和算法库,它们可以帮助开发者编写出更高效的代码,减少CPU和内存的消耗。 2. **跨平台兼容性**:Boost库设计时就考虑了跨平台性,...

    C#移动开发应用实战 使用Mono for Android和.NET C# 中文版

    除此之外,书中还会介绍Android的多线程编程,确保应用的性能和用户体验。还会探讨如何利用Google Play Services集成各种服务,如地图、推送通知和Google Sign-In。最后,会讲解如何调试、测试和发布Android应用,...

    Qt for android实现组播功能

    综上所述,Qt for Android实现组播功能涉及到网络编程、多线程以及跨平台开发等多个方面。通过理解上述概念和步骤,开发者可以有效地构建起能够接收组播数据的应用,从而实现高效的信息分发。在实际项目中,还需要...

    XamarinMobileApplicationDevelopmen for Android英文电子书

    4. **高级主题探索**:进一步探讨了一些高级主题,如多线程编程、性能优化、安全性增强等,帮助读者构建高质量的应用程序。 5. **实战项目案例**:通过一系列实战项目案例,展示了如何将所学知识应用于实际应用开发...

    OpenCV_for_AndroidNDK

    5. **多线程与异步处理**:由于Android设备可能在运行多个任务,因此使用OpenCV进行图像处理时,通常会采用多线程或者异步处理来避免阻塞UI线程。Android的Handler、Thread、AsyncTask或使用NDK的线程API都可以实现...

    android多线程

    在Android开发中,为了提高用户体验并充分利用硬件资源,通常需要实现多线程功能。这样不仅可以避免UI线程被长时间阻塞导致ANR(Application Not Responding)错误,还能有效提升应用性能。 #### 二、在子线程中...

    libvpx Android

    3. 多线程处理:利用Android的多核处理器,将编码任务分配到多个线程中,提高并行处理能力。 总结来说,libvpx Android armeabi_v7a开发包提供了在Android设备上实现高效VP8和VP9视频编码与解码的功能。开发者需要...

    《Xamarin Mobile Application Development for Android》配套源码

    3. **绑定原生库**: Xamarin 提供了绑定 Java 库到 C# 的工具,使得开发者可以利用 Android 平台上的丰富原生库。 **Xamarin Android 应用架构** 1. **活动(Activity)**: 在 Xamarin Android 中,Activity 类是...

Global site tag (gtag.js) - Google Analytics