现在就来学习一下AsyncTask。

一、先介绍一下AsyncTask:

  在开发Android移动客户端的时候往往要使用多线程来进行操作,我们通常会将耗时的操作放在单独的线程执行,避免其占用主线程而给用户带来不好的用户体验。但是在子线程中无法去操作主线程(UI 线程),在子线程中操作UI线程会出现错误。因此android提供了一个类Handler来在子线程中来更新UI线程,用发消息的机制更新UI界面,呈现给用户。这样就解决了子线程更新UI的问题。但是费时的任务操作总会启动一些匿名的子线程,太多的子线程给系统带来巨大的负担,随之带来一些性能问题。因此android提供了一个工具类AsyncTask,顾名思义异步执行任务。这个AsyncTask生来就是处理一些后台的比较耗时的任务,给用户带来良好用户体验的,从编程的语法上显得优雅了许多,不再需要子线程和Handler就可以完成异步操作并且刷新用户界面。

二、构建AsyncTask子类的参数
  AsyncTask<Params,Progress,Result>是一个抽象类
  通常用于被继承,继承AsyncTask需要指定如下三个泛型参数:
  Params:启动任务时输入参数的类型,比如HTTP请求的URL。
  Progress:后台任务执行中返回进度值的类型,后台任务执行的百分比
  Result:后台执行任务完成后返回结果的类型,比如String,Integer等。

三、构建AsyncTask子类的回调方法

  1..execute(Params... params),执行一个异步任务,需要我们在代码中调用此方法,触发异步任务的执行。
  2.onPreExecute:执行后台耗时操作前被调用,通常用户完成一些初始化操作,可以在这里显示进度对话框。

  3、dolnBackground:必须重写,异步执行后台线程将要完成的任务

  4、onPostExecute(),当dolnBackground()完成后,系统自动调用onPostExecute()方法,
  并将dolnBackground()方法的返回值传给该方法,展示处理完后的结果
  5、onProgressUpdata:在dolnBackground()方法中调用pubilshProgress()方法
  更新任务的执行进度后,就回触发该方法。用于显示任务执行的进度。

  6、onCancelled(),在用户取消线程操作的时候调用。在主线程中调用onCancelled()的时候调用。

  几个方法是有规律的。

  (1)首先执行onPreExecute()方法

  (2)执行dolnBackground()方法

  (3)执行onPostExecute()方法

  pubilshProgress()方法则是在dolnBackground()中手动执行,更新任务的进度。

四、几条必须遵守的准则:

  在使用的时候,有几点需要格外注意:

  1.异步任务的实例必须在UI线程中创建。

  2.execute(Params... params)方法必须在UI线程中调用。

  3.不要手动调用onPreExecute(),doInBackground(Params... params),onProgressUpdate(Progress... values),onPostExecute(Result result)这几个方法。

  4.不能在doInBackground(Params... params)中更改UI组件的信息。

  5.一个任务实例只能执行一次,如果执行第二次将会抛出异常

五、简单示例

 一个加载网络图片的效果,加载之前先有3秒的进度显示,然后显示图片

1 package com.example.allcode;
  2 
  3 import java.io.BufferedInputStream;
  4 import java.io.IOException;
  5 import java.io.InputStream;
  6 import java.net.MalformedURLException;
  7 import java.net.URLConnection;
  8 
  9 import android.app.Activity;
 10 import android.graphics.Bitmap;
 11 import android.graphics.BitmapFactory;
 12 import android.os.AsyncTask;
 13 import android.os.Bundle;
 14 import android.util.Log;
 15 import android.view.View;
 16 
 17 import java.net.URL;
 18 import android.widget.ImageView;
 19 import android.widget.ProgressBar;
 20 
 21 public class ImageTest extends Activity{
 22     private AsyncTask as;
 23     private ImageView img_imageview;
 24     private ProgressBar pro;
 25     private static String url="";
 26     @Override
 27     protected void onCreate(Bundle savedInstanceState) {
 28         
 29         
 30         // TODO Auto-generated method stub
 31         super.onCreate(savedInstanceState);
 32         setContentView(R.layout.show_image);
 33         
 34         img_imageview = (ImageView) findViewById(R.id.id_image_show);
 35         pro = (ProgressBar) findViewById(R.id.id_pro);
 36         //设置传递进去的参数
 37         as = new MyAsyncTask().execute(url);
 38     }
 39     //特殊情况,注意点
 40     @Override
 41     protected void onPause() {
 42         // TODO Auto-generated method stub
 43     
 44         super.onPause();
 45         if(as!=null && as.getStatus()==AsyncTask.Status.RUNNING)
 46         {
 47             //只是将对于的AsyncTask标记为cancel状态,并不是真正的取消线程的执行
 48             as.cancel(true);
 49         }
 50     }
 51     //三个参数,URL参数,进度值类型,返回类型
 52     class MyAsyncTask extends AsyncTask<String, Void, Bitmap>{
 53 
 54     //必须实现的方法
 55     @Override
 56     //,异步执行后台线程将要完成的任务
 57     protected Bitmap doInBackground(String... params) {
 58         
 59         // TODO Auto-generated method stub
 60         //传进来的是一个可变长数组,不止一个参数,只需要取出第0位
 61         String url = params[0];
 62         //访问网络的操作
 63         Bitmap bitmap = null;
 64         URLConnection connection;
 65         InputStream is;
 66         
 67         try {
 68             connection  = new URL(url).openConnection(); //获取网络连接对象
 69             is = connection.getInputStream(); //获取输入流
 70             BufferedInputStream bis = new BufferedInputStream(is);
 71             Thread.sleep(3000);//睡眠三秒钟,方便查看效果
 72             bitmap = BitmapFactory.decodeStream(bis);//将输入流解析成bitmap
 73             is.close();//关闭输入流
 74         } catch (MalformedURLException e) {
 75             // TODO Auto-generated catch block
 76             e.printStackTrace();
 77         } catch (IOException e) {
 78             // TODO Auto-generated catch block
 79             e.printStackTrace();
 80         } catch (InterruptedException e) {
 81             // TODO Auto-generated catch block
 82             e.printStackTrace();
 83         }
 84         
 85         //publishProgress();
 86         //返回bitmap,作为返回值
 87         return bitmap;
 88     }
 89     //手动添加其他方法
 90     //执行后台耗时操作前被调用,通常用户完成一些初始化操作,可以在这里显示进度对话框。
 91     @Override
 92     protected void onPreExecute() {
 93         // TODO Auto-generated method stub
 94         
 95         super.onPreExecute();
 96         pro.setVisibility(View.VISIBLE);//将进度条显示出来
 97     }
 98     //当dolnBackground()完成后,系统自动调用onPostExecute()方法,
 99     //  并将dolnBackground()方法的返回值传给该方法,展示处理完后的结果
100     //这里操作UI,设置图像
101     @Override
102     protected void onPostExecute(Bitmap bitmap) {
103         // TODO Auto-generated method stub
104         
105         super.onPostExecute(bitmap);
106         pro.setVisibility(View.GONE);//隐藏进度条
107         img_imageview.setImageBitmap(bitmap); //显示图片
108     }
109     
110     @Override
111     protected void onProgressUpdate(Void... values) {
112         // TODO Auto-generated method stub
113         
114         super.onProgressUpdate(values);
115         Log.d("xyd","onProgressUpdate");
116     }
117 }
118 
119 }



对应的布局文件:


1 <?xml version="1.0" encoding="utf-8"?>
 2 <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
 3     android:layout_width="match_parent"
 4     android:layout_height="match_parent"
 5     android:orientation="vertical" >
 6     
 7     <ImageView 
 8         android:id="@+id/id_image_show"
 9         android:layout_height="wrap_content"
10         android:layout_width="wrap_content"
11         
12         />
13     <ProgressBar 
14         android:id="@+id/id_pro"
15         android:visibility="gone"
16         android:layout_centerInParent="true"
17         android:layout_width="wrap_content"
18         android:layout_height="wrap_content"
19         />
20 </RelativeLayout>


注意在AndroidManifest.xml文件中加上网络权限

<uses-permission android:name="android.permission.INTERNET"/>