Android多线程分析之一:使用Thread异步下载图像

打算整理一下对 Android Framework 中多线程相关知识的理解,主要集中在 Framework 层的 Thread, Handler, Looper, MessageQueue, Message, AysncTask,当然不可避免地要涉及到 native 方法,因此也会分析 dalvik 中和线程以及消息处理相关的代码:如 dalvik 中的 C++ Thread 类以及 MessageQueue 类。本文将从一个使用 Thread 的简单 应用入手,引入 Thread 这个话题,接下来的几篇文章会依次介绍前面提到的那些主题。


这是一个使用 Android Thread 从网络上异步下载图片并在 ImageView 中显示的的简单示例。因为需要访问网络,所以要在 manifest.xml 中添加网络访问权限:

[html]  view plain  copy
 print ?
  1. <uses-permission android:name="android.permission.INTERNET">  
  2. uses-permission>  

布局文件很简单,一个 Button,一个 ImageView:

[html]  view plain  copy
 print ?
  1. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  2.     xmlns:tools="http://schemas.android.com/tools"  
  3.     android:layout_width="match_parent"  
  4.     android:layout_height="match_parent"  
  5.     android:orientation="vertical"  
  6.     android:padding="10dip" >  
  7.   
  8.     <Button  
  9.         android:id="@+id/LoadButton"  
  10.         android:layout_width="fill_parent"  
  11.         android:layout_height="wrap_content"  
  12.         android:text="Load">  
  13.     Button>  
  14.   
  15.     <ImageView  
  16.         android:id="@+id/ImageVivew"   
  17.         android:layout_width="match_parent"   
  18.         android:layout_height="400dip"   
  19.         android:scaleType="centerInside"   
  20.         android:padding="2dp">  
  21.     ImageView>   
  22.       
  23. LinearLayout>  

接下来看代码:

首先来看定义:图片的 url 路径,两个消息值以及一些控件:

[java]  view plain  copy
 print ?
  1.    private static final String sImageUrl = "http://fashion.qqread.com/ArtImage/20110225/0083_13.jpg";  
  2.   
  3. private static final int MSG_LOAD_SUCCESS = 0;  
  4. private static final int MSG_LOAD_FAILURE = 1;  
  5.   
  6.    private Button mLoadButton;  
  7.    private ProgressDialog mProgressBar;  
  8.    private ImageView mImageView;  

然后来看控件的设置:

[java]  view plain  copy
 print ?
  1. protected void onCreate(Bundle savedInstanceState) {  
  2.     super.onCreate(savedInstanceState);  
  3.     setContentView(R.layout.activity_main);  
  4.       
  5.     Log.i("UI thread"" >> onCreate()");  
  6.       
  7.     mProgressBar = new ProgressDialog(this);  
  8.     mProgressBar.setCancelable(true);  
  9.     mProgressBar.setMessage("Image downloading ...");  
  10.     mProgressBar.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);  
  11.     mProgressBar.setMax(100);  
  12.       
  13.     mImageView = (ImageView)this.findViewById(R.id.ImageVivew);  
  14.       
  15.     mLoadButton = (Button)this.findViewById(R.id.LoadButton);  
  16.     mLoadButton.setOnClickListener(new View.OnClickListener() {  
  17.            @Override   
  18.            public void onClick(View v) {  
  19.             mProgressBar.setProgress(0);  
  20.             mProgressBar.show();  
  21.               
  22.             new Thread() {  
  23.                 @Override  
  24.                 public void run() {  
  25.                     Log.i("Load thread"" >> run()");  
  26.                        Bitmap bitmap = loadImageFromUrl(sImageUrl);  
  27.                        if (bitmap != null) {  
  28.                         Message msg = mHandler.obtainMessage(MSG_LOAD_SUCCESS, bitmap);  
  29.                         mHandler.sendMessage(msg);  
  30.                        }  
  31.                        else {  
  32.                         Message msg = mHandler.obtainMessage(MSG_LOAD_FAILURE, null);  
  33.                         mHandler.sendMessage(msg);  
  34.                        }  
  35.                 }  
  36.             }.start();  
  37.            }  
  38.        });  
  39. }  

loadImageFromUrl 是一个从网络下载 Bitmap 的 static 函数:

[java]  view plain  copy
 print ?
  1. static Bitmap loadImageFromUrl(String uil) {  
  2.     Bitmap bitmap = null;  
  3.     try{  
  4.         InputStream in = new java.net.URL(sImageUrl).openStream();  
  5.         bitmap = BitmapFactory.decodeStream(in);  
  6.         in.close();  
  7.     }  
  8.     catch (Exception e) {  
  9.         e.printStackTrace();  
  10.     }  
  11.     return bitmap;  
  12. }  

mHandler 是主线程也就是 UI 线程处理消息的 Handler:

[java]  view plain  copy
 print ?
  1. private Handler mHandler= new Handler(){  
  2.     @Override  
  3.     public void handleMessage(Message msg) {  
  4.         Log.i("UI thread"" >> handleMessage()");  
  5.           
  6.         switch(msg.what){  
  7.         case MSG_LOAD_SUCCESS:  
  8.             Bitmap bitmap = (Bitmap) msg.obj;  
  9.             mImageView.setImageBitmap(bitmap);  
  10.               
  11.             mProgressBar.setProgress(100);  
  12.             mProgressBar.setMessage("Image downloading success!");  
  13.             mProgressBar.dismiss();  
  14.             break;  
  15.               
  16.         case MSG_LOAD_FAILURE:  
  17.             mProgressBar.setMessage("Image downloading failure!");  
  18.             mProgressBar.dismiss();  
  19.             break;  
  20.         }  
  21.     }  
  22. };  

纵观上面的代码,当点击 load 按钮时,会创建一个匿名 Thread,并调用其 start() 启动运行线程,在这个线程中进行图像下载并解码成 Bitmap,然后通过 Handler 向 UI 线程发送消息以通知下载结果。这都是在匿名 Thead 中处理的。主线程也就是 UI 线程收到消息之后,会分发给 Handler,在它的 handleMessage 方法中根据消息 id 来处理下载结果,要么成功要么失败,并相应地更新 UI。


运行该示例:

Android多线程分析之一:使用Thread异步下载图像_第1张图片

可以从 logcat 的第四栏看到 UI thread(tid: 817) 和 Load thread(tid: 830) 的线 程id 是不同的,因为它们是两个独立的线程。


在匿名线程下载完毕之后,为什么不直接在这个线程的 run() 中更新 UI 呢?这样做有什么后果?这些问题将在后文详细解答。


你可能感兴趣的:(android)