AsyncTask介绍

AsyncTask比Handler更轻量级一些,适用于简单的异步处理。

使用AsyncTask时,注意重写以下几个方法:

1. doInBackground()

作用:执行后台任务。
要求:必须重写!
注意:在doInBackground()中不能进行UI操作!

2. onPreExecute()

作用:通常用于执行后台任务开始前的准备动作。在用户调用execute()后,并且在AsyncTask主动执行doInBackground()之前被调用。
要求:选择性重写。如果不重写该函数,默认不执行任何动作

3. onPostExecute()

作用:通常用于处理后台动作的返回结果。在AsyncTask主动执行doInBackground()之前被调用。
要求:选择性重写。如果不重写该函数,默认不执行任何动作

4. onProgressUpdate()

作用:通常用于执行后台任务执行期间的进度更新。因为doInBackground()中不能操作UI,假如我们想在后台任务处理时显示进度,可以在doInBackground()中调用publishProgress(),而publicProgress()会调用onProgressUpdate();在onProgressUpdate()中进行UI操作即可。
要求:选择性重写。如果不重写该函数,默认不执行任何动作

5. onCancelled()

作用:通常用于执行取消AsyncTask任务时的相关动作。如果客户主动调用cancel(),则会执行onCancelled();否则(AsyncTask执行完之后正常停止),则不会调用onCancelled()。
要求:选择性重写。如果不重写该函数,默认不执行任何动作

AsyncTask使用和示例

下面是一个自定义的AsyncTask。

  1. private class MyTask extends AsyncTask<String, Integer, String> {
  2. //onPreExecute方法用于在执行后台任务前做一些UI操作
  3. @Override
  4. protected void onPreExecute() {
  5. Log.i(TAG, "onPreExecute");
  6. textView.setText("loading...");
  7. }
  8. //doInBackground方法内部执行后台任务,不可在此方法内修改UI
  9. @Override
  10. protected String doInBackground(String... params) {
  11. Log.i(TAG, "doInBackground");
  12. try {
  13. for (int i=0; i<6; i++) {
  14. Log.d(TAG, "doInBackground: publishProgress="+i);
  15. publishProgress(20*i);
  16. Thread.sleep(500);
  17. }
  18. Log.d(TAG, "doInBackground: return OK!");
  19. return "OK";
  20. } catch (InterruptedException e) {
  21. e.printStackTrace();
  22. }
  23. Log.d(TAG, "doInBackground: return FAIL!");
  24. return "FAIL";
  25. }
  26. //onProgressUpdate方法用于更新进度信息
  27. @Override
  28. protected void onProgressUpdate(Integer... progresses) {
  29. Log.i(TAG, "onProgressUpdate");
  30. progressBar.setProgress(progresses[0]);
  31. textView.setText("loading..." + progresses[0] + "%");
  32. }
  33. //onPostExecute方法用于在执行完后台任务后更新UI,显示结果
  34. @Override
  35. protected void onPostExecute(String result) {
  36. Log.i(TAG, "onPostExecute");
  37. textView.setText(result);
  38. execute.setEnabled(true);
  39. cancel.setEnabled(false);
  40. }
  41. //onCancelled方法用于在取消执行中的任务时更改UI
  42. @Override
  43. protected void onCancelled() {
  44. Log.i(TAG, "onCancelled");
  45. textView.setText("cancelled");
  46. progressBar.setProgress(0);
  47. execute.setEnabled(true);
  48. cancel.setEnabled(false);
  49. }
  50. }

说明:
(01) textView是一个TextView对象,cancel和execute分别是两个Button按钮,而progressBar则是进度条。
(02) 在任务开始时会通过onPreExecute()更新TextView的显示内容。
(03) 在任务结束时通过onPreExecute()更新TextView的显示内容。
(04) 任务执行过程中会通过publishProgress()更新进度条。
(05) 如果任务被强制取消的话,会将进度条重置为0。

创建并执行AsyncTask的接口如下:

  1. mTask = new MyTask();
  2. mTask.execute("http://www.baidu.com");

取消AsyncTask的接口如下:

  1. mTask.cancel();

点击查看:AsyncTask示例完整原理

AsyncTask原理

下面通过Android4.4.2的AsyncTask源码来对AsyncTask原理进行介绍。

1. AsyncTask构造函数

  1. public AsyncTask() {
  2. mWorker = new WorkerRunnable<Params, Result>() {
  3. public Result call() throws Exception {
  4. mTaskInvoked.set(true);
  5. Process.setThreadPriority(Process.THREAD_PRIORITY_BACKGROUND);
  6. //noinspection unchecked
  7. return postResult(doInBackground(mParams));
  8. }
  9. };
  10. mFuture = new FutureTask<Result>(mWorker) {
  11. @Override
  12. protected void done() {
  13. try {
  14. postResultIfNotInvoked(get());
  15. } catch (InterruptedException e) {
  16. android.util.Log.w(LOG_TAG, e);
  17. } catch (ExecutionException e) {
  18. throw new RuntimeException("An error occured while executing doInBackground()",
  19. e.getCause());
  20. } catch (CancellationException e) {
  21. postResultIfNotInvoked(null);
  22. }
  23. }
  24. };
  25. }

说明:

(01) mWorker是个WorkerRunnable对象,而WorkerRunnable是Callable的实现类。而在"线程池中关于Callable的介绍"时,我们说过,Callable类似于Runnable接口,不同之处主要在于Callable能和Future配合使用获取任务的结果,而Runnable不能获取结果!
(02) mFuture是FutureTask对象,而FutureTask洽洽是Future的实现类。 mWorker和mFuture配合使用,能获取后台任务的结果!

2. execute

execute()的源码如下:

  1. public final AsyncTask<Params, Progress, Result> execute(Params... params) {
  2. return executeOnExecutor(sDefaultExecutor, params);
  3. }

说明:sDefaultExecutor是线程池对象,而executeOnExecutor()是真正执行后台动作的地方。

下面是线程池相关的代码:

  1. private static final int CORE_POOL_SIZE = CPU_COUNT + 1;
  2. private static final int MAXIMUM_POOL_SIZE = CPU_COUNT * 2 + 1;
  3. private static final int KEEP_ALIVE = 1;
  4. public static final Executor THREAD_POOL_EXECUTOR
  5. = new ThreadPoolExecutor(CORE_POOL_SIZE, MAXIMUM_POOL_SIZE, KEEP_ALIVE,
  6. TimeUnit.SECONDS, sPoolWorkQueue, sThreadFactory);
  7. ...
  8. public static final Executor SERIAL_EXECUTOR = new SerialExecutor();
  9. private static volatile Executor sDefaultExecutor = SERIAL_EXECUTOR;
  10. private static class SerialExecutor implements Executor {
  11. final ArrayDeque<Runnable> mTasks = new ArrayDeque<Runnable>();
  12. Runnable mActive;
  13. public synchronized void execute(final Runnable r) {
  14. mTasks.offer(new Runnable() {
  15. public void run() {
  16. try {
  17. r.run();
  18. } finally {
  19. scheduleNext();
  20. }
  21. }
  22. });
  23. if (mActive == null) {
  24. scheduleNext();
  25. }
  26. }
  27. protected synchronized void scheduleNext() {
  28. if ((mActive = mTasks.poll()) != null) {
  29. THREAD_POOL_EXECUTOR.execute(mActive);
  30. }
  31. }
  32. }

说明:该线程池是"通过双向队列实现的串行线程池"。scheduleNext()每次只会执行一个任务。

下面看看executeOnExecutor()的代码:

  1. public final AsyncTask<Params, Progress, Result> executeOnExecutor(Executor exec,
  2. Params... params) {
  3. if (mStatus != Status.PENDING) {
  4. switch (mStatus) {
  5. case RUNNING:
  6. throw new IllegalStateException("Cannot execute task:"
  7. + " the task is already running.");
  8. case FINISHED:
  9. throw new IllegalStateException("Cannot execute task:"
  10. + " the task has already been executed "
  11. + "(a task can be executed only once)");
  12. }
  13. }
  14. mStatus = Status.RUNNING;
  15. onPreExecute();
  16. mWorker.mParams = params;
  17. exec.execute(mFuture);
  18. return this;
  19. }

说明:
(01) executeOnExecutor()首先会对任务的状态进行处理。任务共三种姿态:

PENDING: 挂起状态。当AsyncTask被创建时,就进入了PENDING状态。
RUNNING: 运行状态。当AsyncTask被执行时,就进入了RUNNING状态。
FINISHED: 完成状态。当AsyncTask完成(被客户cancel()或正常运行完毕)时,就进入了FINISHED状态。

当任务是RUNNING或PENDING状态时,会抛出异常。这就决定了,一个AsyncTask只能被执行一次,即只能对一个AsyncTask调用一次execute();如果要重新执行任务,则需要新建AsyncTask后再调用execute()。
(02) 接着,调用onPreExecute()。这也就是任务执行前的准备动作!
(03)
然后,调用exec.execute(mFuture)。作用是将任务提交到线程池中进行执行。线程池的代码前面已经给出,SerialExecutor中的execute()会执行r.run()任务。r.run()实际上是调用FutureTask中run()方法,而FutureTask的run()方法,则会执行Callable的call()函数,即会执行到mWorker的call()方法。而观察前面mWorker的run()方法,我们会发现它会调用doInBackground()接口,并通过postResult()返回任务执行结果。而postResult()的内容如下:

  1. private static final InternalHandler sHandler = new InternalHandler();
  2. private Result postResult(Result result) {
  3. @SuppressWarnings("unchecked")
  4. Message message = sHandler.obtainMessage(MESSAGE_POST_RESULT,
  5. new AsyncTaskResult<Result>(this, result));
  6. message.sendToTarget();
  7. return result;
  8. }
  9. private static class InternalHandler extends Handler {
  10. @SuppressWarnings({"unchecked", "RawUseOfParameterizedType"})
  11. @Override
  12. public void handleMessage(Message msg) {
  13. AsyncTaskResult result = (AsyncTaskResult) msg.obj;
  14. switch (msg.what) {
  15. case MESSAGE_POST_RESULT:
  16. // There is only one result
  17. result.mTask.finish(result.mData[0]);
  18. break;
  19. case MESSAGE_POST_PROGRESS:
  20. result.mTask.onProgressUpdate(result.mData);
  21. break;
  22. }
  23. }
  24. }

说明:postResult()会发送MESSAGE_POST_RESULT给sHandler,而sHandler中会将任务执行结果传递给finish()。以下是finish()的代码:

  1. private void finish(Result result) {
  2. if (isCancelled()) {
  3. onCancelled(result);
  4. } else {
  5. onPostExecute(result);
  6. }
  7. mStatus = Status.FINISHED;
  8. }

说明:如果是正常执行结束,则调用onPostExecute()方法;否则(异常结束),则调用onCancelled()方法。

至此,AsyncTask的原理结果完毕!总的来说,就是通过线程池来实现的,AsyncTask的任务会提交到线程池中,执行完后,线程池再返回结果。

ps:摘录自网络

AsyncTask介绍的更多相关文章

  1. Android笔记——AsyncTask介绍

    AsyncTask和Handler对比 1 ) AsyncTask实现的原理,和适用的优缺点 AsyncTask,是android提供的轻量级的异步类,可以直接继承AsyncTask,在类中实现异步操 ...

  2. 55.Android之AsyncTask介绍 (转)

    AsyncTask和Handler对比 1 ) AsyncTask实现的原理,和适用的优缺点 AsyncTask,是android提供的轻量级的异步类,可以直接继承AsyncTask,在类中实现异步操 ...

  3. android AsyncTask介绍(转)

    android AsyncTask介绍 AsyncTask和Handler对比 1 ) AsyncTask实现的原理,和适用的优缺点 AsyncTask,是android提供的轻量级的异步类,可以直接 ...

  4. android AsyncTask介绍 转载

    http://www.cnblogs.com/devinzhang/archive/2012/02/13/2350070.html AsyncTask和Handler对比 1 ) AsyncTask实 ...

  5. android AsyncTask介绍 AsyncTask和Handler对比

    1 ) AsyncTask实现的原理,和适用的优缺点 AsyncTask,是android提供的轻量级的异步类,可以直接继承AsyncTask,在类中实现异步操作,并提供接口反馈当前异步执行的程度(可 ...

  6. android AsyncTask介绍

    AsyncTask是android提供的轻量级的异步类,可以直接继承AsyncTask,在类中实现异步操作,并提供接口反馈当前异步执行的程度(可以通过接口实现UI进度更新),最后反馈执行的结果给UI主 ...

  7. android:异步任务asyncTask介绍及异步任务下载图片(带进度条)

    为什么要用异步任务? 在android中仅仅有在主线程才干对ui进行更新操作.而其他线程不能直接对ui进行操作 android本身是一个多线程的操作系统,我们不能把全部的操作都放在主线程中操作 .比方 ...

  8. 同步任务 AsyncTask 介绍

    AsyncTask 顾名思义,是在我们需要执行同步任务的时候使用,这个类可以做一些后台操作,然后将结果返回的UI来,因为这个类本身封装了Handler和Thread,所以我们不需要直接去操作这两个类, ...

  9. AsyncTask的介绍

    android AsyncTask介绍 AsyncTask和Handler对比 1 ) AsyncTask实现的原理,和适用的优缺点 AsyncTask,是android提供的轻量级的异步类,可以直接 ...

随机推荐

  1. C#:调用webservice时提示对操作的回复消息正文进行反序列化时出错

    主要原因webservice返回值的长度超过readerQuotas中的了maxStringContentLength值,造成返回值截断,不完整,反序列化时出错. <readerQuotas m ...

  2. GBDT算法原理深入解析

    GBDT算法原理深入解析 标签: 机器学习 集成学习 GBM GBDT XGBoost 梯度提升(Gradient boosting)是一种用于回归.分类和排序任务的机器学习技术,属于Boosting ...

  3. jQuery源码:从原理到实战

    jQuery源码:从原理到实战 jQuery选择器对象 $(".my-class"); document.querySelectorAll*".my-class" ...

  4. 错误: 未能从 xmlsocket://127.0.0.1:5840 中加载策略文件

    看看你是否使用了MonsterDebugger,如果是这样的话, 因为那个 MonsterDebugger 没有启动 删掉MonsterDebugger的代码吧

  5. [HTML5]HTML结构性元素(Structure)

    参考自:http://techbrood.com/h5b2a?p=html-structure 结构性元素用来组织文档的各个部分 为了让文档层次分明,我们可以把文档中的元素按其内容的作用进行组合,这就 ...

  6. ArcGIS Engine渲染

    符号化之Renderer( 渲染)体系 ArcGIS Engine9.3对GIS数据的符号化分为矢量数据渲染和栅格数据渲染两大类.接下来分别介绍FeatureRender和RasterRender. ...

  7. Google Font字体本地化使用提高网站访问速度

    Google Web font在国内经常不稳定,速度在国内延迟也很高,而引发网页打开速度慢. 一.常见的字体格式介绍 不同的浏览器对字体格式支持是不一致的,常见的如下: 1.TureTpe(.ttf) ...

  8. Unity3D 动画回调方法

    最近发现很多coder.在用Unity开发游戏的时候都需要一个需求就是..动画播到某一帧就要干什么事情.而且希望能得到回调. 在unity里面的window菜单有个.Animation工具.打开它.然 ...

  9. AE开发示例之GPBufferLayer

    using System; using System.Drawing;using System.Text;using System.Windows.Forms;using System.Runtime ...

  10. 关于oracle中创建新表时将我们要用的表的结构和数据都复制过去

    今天在oracle中遇到了一个问题,就是给我查询出来了一张表的数据,只有部分的字段,让我将这张表的结构和数据放到新的临时表中,并进行数据的查询. 我是这样做的: 如:create table tabl ...