概述

Android 中只有主线程可以操作 ui,以保证 ui 的稳定性,而 Android 是多线程的操作系统,像读取文件、网络操作等耗时较长的操作应该放到子线程中执行,以免阻塞主线程。

利用 AsyncTask 可方便的实现异步操作,既可以在子线程中执行异步操作,又可以更新 ui。

AsyncTask<Params, Progress, Result>是一个抽象类,用于被继承以实现自定义的异步任务。

继承 AsyncTask 需要指定三个参数类型:

  • Params:启动任务时输入参数的类型。
  • Progress:后台任务执行中返回进度值的类型。
  • Result:后台执行任务完成后返回结果的类型。

AsyncTask 通常需要重写的方法有四个:

  • onPreExecute:执行后台耗时操作前被调用,通常用于完成一些初始化操作。运行在主线程。
  • doInBackground:必须重写。异步执行后台线程将要完成的任务。
  • onProgressUpdate:在doInBackground方法中调用publishProgress方法时调用,用于更新进度。运行在主线程。
  • onPostExecute:当doInBackground完成后,系统会将doInBackground的返回值传入该方法,以进行后续操作。运行在主线程。

例子:加载网络图片

activity_load_image.xml
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http:///apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:padding="16dp">

    <ImageView
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:id="@+id/image"/>

    <ProgressBar
        android:visibility="gone"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_centerInParent="true"
        android:id="@+id/progressbar"/>

</RelativeLayout>
LoadImageActivity.java
public class LoadImageActivity extends Activity {

    /**
     * 网络图片的地址
     */
    private static final String URL = "http://imgsrc.baidu.com/forum/w=580/sign=57ae714ad2160924dc25a213e407359b/8a82b9014a90f60392de8a5b3b12b31bb051ed44.jpg";

    private ImageView imageView;
    private ProgressBar progressBar;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_load_image);

        imageView = (ImageView) findViewById(.image);
        progressBar = (ProgressBar) findViewById(.progressbar);

        //启动异步任务
        new LoadImageAsyncTask().execute(URL);
    }

    class LoadImageAsyncTask extends AsyncTask<String, Void, Bitmap> {

        /**
         * 异步任务的准备工作(运行在ui线程)
         */
        @Override
        protected void onPreExecute() {
            super.onPreExecute();
            //显示进度条
            progressBar.setVisibility(View.VISIBLE);
        }

        /**
         * 异步任务(运行在异步线程)
         * @param strings
         * @return
         */
        @Override
        protected Bitmap doInBackground(String... strings) {
            String url = strings[0];
            Bitmap bitmap = null;
            if (url != null && url.isEmpty() == false) {
                //下载网络图片,转换为Bitmap
                URLConnection urlConnection = null;
                InputStream is = null;

                try {
                    urlConnection = new URL(url).openConnection();
                    is = urlConnection.getInputStream();

                    BufferedInputStream bis = new BufferedInputStream(is);
                    bitmap = BitmapFactory.decodeStream(bis);

                    is.close();
                    bis.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            return bitmap;
        }

        /**
         * 异步任务的结果处理(运行在ui线程)
         * @param bitmap
         */
        @Override
        protected void onPostExecute(Bitmap bitmap) {
            super.onPostExecute(bitmap);
            //隐藏进度条
            progressBar.setVisibility(View.GONE);
            //显示图片
            if (bitmap != null) {
                imageView.setImageBitmap(bitmap);
            }
        }
    }
}
AndroidManifest.xml
<uses-permission android:name="android.permission.INTERNET"/>

例子:更新进度条 & 取消异步任务

activity_load_progressbar.xml
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http:///apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:padding="16dp">

    <ProgressBar
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_centerInParent="true"
        android:id="@+id/progressbar"
        style="?android:attr/progressBarStyleHorizontal"/>

</RelativeLayout>
LoadProgressbarActivity.java
public class LoadProgressbarActivity extends Activity {

    private ProgressBar progressBar;
    private LoadProgressbarAsyncTask asyncTask;


    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_load_progressbar);

        progressBar = (ProgressBar) findViewById(.progressbar);
        asyncTask = new LoadProgressbarAsyncTask();
        asyncTask.execute();
    }

    @Override
    protected void onPause() {
        if (asyncTask != null && asyncTask.getStatus() == AsyncTask.Status.RUNNING) {
            //标记为取消状态,否则在退出activity再进入activity时,要等待前一个任务执行完成才会开始执行
            asyncTask.cancel(true);
        }

        super.onPause();
    }

    class LoadProgressbarAsyncTask extends AsyncTask<Void, Integer, Void> {

        @Override
        protected Void doInBackground(Void... voids) {
            for (int i=0; i<100; i++) {
                //判断若已取消,则停止
                if (isCancelled()) {
                    break;
                }

                try {
                    Thread.sleep(300);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }

                //更新进度
                publishProgress(i);
            }
            return null;
        }

        @Override
        protected void onProgressUpdate(Integer... values) {
            super.onProgressUpdate(values);

            if (isCancelled()) {
                return;
            }

            //更新进度条
            progressBar.setProgress(values[0]);
        }
    }
}

AsyncTask 注意事项

  • 必须在 ui 线程中创建 AsyncTask 的实例。
  • 必须在 ui 线程中调用 execute 方法。
  • 重写的四个方法是系统自动调用的,不应手动调用。
  • 每个 AsyncTask 只能被执行一次,多次调用会引发异常。
  • doInBackground 中不能直接更新 ui。

参考:http://www.imooc.com/learn/377