Quantcast
Channel: CSDN博客推荐文章
Viewing all articles
Browse latest Browse all 35570

Android AsyncTask源码分析

$
0
0

AsyncTask是一个抽象类,我们需要继承这个类,并实现抽象方法,在这个类的实现过程中需要定义3个类和执行4个方法。3个类分别是:

1.Params:执行异步任务需要的参数

2.Progress:异步任务后台执行中需要刷新的参数

3.Result:后台任务执行完毕需要返回的结果

当一个异步的任务执行,会执行4个方法,分别是:

1.              onPreExecute():在异步任务调用之前,由UI线程中调用。通常用于异步任务的准备工作。

2.              doinBackground(Params …):在上一步执行完毕后立即执行。是比较耗时的一步,也是由这步传入参数Params。在这里不仅可以产生最后的结果result,而且能调用publishProgress(Progress...)刷新UI

3.              onProgressUpdate(Progress...): 在第二步中调用publishProgress()之后,UI线程则会调用此方法。举个例子,就像你先在异步的线程中传入UI线程的handler,当执行的过程中,你可以通过消息机制向主线程发送消息,来更新UI。而publishProgress就是发送消息,主线程就会调用onProgressUpdate

4.              onPostExecute(Result): 当异步的任务执行完毕之后,由UI线程调用。在这个方法中可以得到最终的异步结果。

 本文主要讲解这4个回调函数的调用流程,

首先看构造函数:

    public AsyncTask() {
        mWorker = new WorkerRunnable<Params, Result>() {
            public Result call() throws Exception {
                mTaskInvoked.set(true);

                Process.setThreadPriority(Process.THREAD_PRIORITY_BACKGROUND);
                return postResult(doInBackground(mParams));
            }
        };

        mFuture = new FutureTask<Result>(mWorker) {
            @Override
            protected void done() {
                try {
                    final Result result = get();

                    postResultIfNotInvoked(result);
                } catch (InterruptedException e) {
                    android.util.Log.w(LOG_TAG, e);
                } catch (ExecutionException e) {
                    throw new RuntimeException("An error occured while executing doInBackground()",
                            e.getCause());
                } catch (CancellationException e) {
                    postResultIfNotInvoked(null);
                } catch (Throwable t) {
                    throw new RuntimeException("An error occured while executing "
                            + "doInBackground()", t);
                }
            }
        };
    }

构造函数里面创建了2个对象,mWorker和mFuture。mWorker是一个Callable,mWorker实现了其call( ) 方法。

mFuture是一个FutureTask,这个FutureTask使用Callable作为构造函数的参数,并实现了done( )方法。

再来看看AsyncTask的execure(Param...)方法:

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

    public final AsyncTask<Params, Progress, Result> executeOnExecutor(Executor exec,
            Params... params) {
        if (mStatus != Status.PENDING) {
            switch (mStatus) {
                case RUNNING:
                    throw new IllegalStateException("Cannot execute task:"
                            + " the task is already running.");
                case FINISHED:
                    throw new IllegalStateException("Cannot execute task:"
                            + " the task has already been executed "
                            + "(a task can be executed only once)");
            }
        }

        mStatus = Status.RUNNING;

        onPreExecute();

        mWorker.mParams = params;
        exec.execute(mFuture);

        return this;
    }
现在从execute开始分析:

1、execute调用executeOnExecutor

2、executeOnExecutor调用onPreExecute

3、随后执行:exec.execute(mFuture)

4、mFuture被执行后,mWorker的call()开始执行

5、call()里面调用doInBackground()

6、doInBackground()执行完成后,马上执行postResult()

7、postResult导致result.mTask.finish(result.mData[0])执行

8、finish()会调用onPostExecute()

至此,4个回调方法中的3个已经讲完了。

还剩下onProgressUpdate(),其实onProgressUpdate()的执行是被动的,所以不在上面的分析之中,它的执行需要app去调用publishProgress。

app一般会在doInBackground()中的循环中执行publishProgress,这样就回调到onProgressUpdate,而onProgressUpdate()一般被实现为更新UI的函数。
至此4个函数都分析到了,如果读者对Callable和FutureTask不是很理解,请阅读我的上一篇文章:Java FutureTask理解

作者:gemmem 发表于2013-5-21 20:18:40 原文链接
阅读:56 评论:0 查看评论

Viewing all articles
Browse latest Browse all 35570

Trending Articles



<script src="https://jsc.adskeeper.com/r/s/rssing.com.1596347.js" async> </script>