屡思路
1. 初始 AsyncTask
AsyncTask这个类的声明如下:
2
3
4
public abstract class AsyncTask<Params, Progress, Result> {
.....
}
是一个抽象类Params 表示输入参数的类型Progress 表示后台任务的执行进度Result 表示返回结果的类型
2.使用
在
AsyncTask这个类的顶部有一些代码注释,里面讲述了如何使用一个AsyncTask,如下:
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
* <pre class="prettyprint">
* private class DownloadFilesTask extends AsyncTask<URL, Integer, Long> {
* protected Long doInBackground(URL... urls) {
* int count = urls.length;
* long totalSize = 0;
* for (int i = 0; i < count; i++) {
* totalSize += Downloader.downloadFile(urls[i]);
* publishProgress((int) ((i / (float) count) * 100));
* // Escape early if cancel() is called
* if (isCancelled()) break;
* }
* return totalSize;
* }
*
* protected void onProgressUpdate(Integer... progress) {
* setProgressPercent(progress[0]);
* }
*
* protected void onPostExecute(Long result) {
* showDialog("Downloaded " + result + " bytes");
* }
* }
* </pre>
//user
<p>Once created, a task is executed very simply:</p>
* <pre class="prettyprint">
* new DownloadFilesTask().execute(url1, url2, url3);
* </pre>
3. 内部重要方法
- onPreExecute()- 1 
 2
 3- @MainThread 
 protected void onPreExecute() {
 }
在主线程中运行,异步任务之前会被调用,一般用于做一些准备工作;
- doInBackground()- 1 
 2- @WorkerThread 
 protected abstract Result doInBackground(Params... params);
在线程池中运行,此方法一般用于执行异步任务,通过publishProgress 方法来更新进度;
- onProgressUpdate()- 1 
 2
 3- @MainThread 
 protected void onProgressUpdate(Progress... values) {
 }
主线程中运行,当通过publishProgress 方法调用后,onProgressUpdate() 方法会被调用;
- onPostExecute()- 1 
 2
 3- @MainThread 
 protected void onPostExecute(Result result) {
 }
主线程中运行,将返回的结果展示。
4.源码分析
从它的 execute 方法开始:1
2
3
4
5
6
7
8 @MainThread
    public final AsyncTask<Params, Progress, Result> execute(Params... params) {
      //sDefaultExecutor 定义如下,线程池
        return executeOnExecutor(sDefaultExecutor, params);
    }
private static volatile Executor sDefaultExecutor = SERIAL_EXECUTOR;
public static final Executor SERIAL_EXECUTOR = new SerialExecutor();
| 1 | @MainThread | 
为了弄明白整体流程,首页要搞明白上面的 mWorker  mFuture 是干嘛的。
- mWorker
| 1 | private final WorkerRunnable<Params, Result> mWorker; | 
| 1 | //在AsyncTask 的构造方法中,分别对 mWorker, mFuture 进行了初始化 | 
postResult(result) 方法1
2
3
4
5
6
7private Result postResult(Result result) {
       @SuppressWarnings("unchecked")
       Message message = getHandler().obtainMessage(MESSAGE_POST_RESULT,
               new AsyncTaskResult<Result>(this, result));
       message.sendToTarget();
       return result;
   }
发送一条 MESSAGE_POST_RESULT 的消息,并且将result 存入到了 AsyncTaskResult中的 mData 中去了,
其中 AsyncTaskResult1
2
3
4
5
6
7
8
9
10@SuppressWarnings({"RawUseOfParameterizedType"})
    private static class AsyncTaskResult<Data> {
        final AsyncTask mTask;
        final Data[] mData;
        AsyncTaskResult(AsyncTask task, Data... data) {
            mTask = task;
            mData = data;
        }
    }
getHandler 获取一个 Handler ,我们看下 handleMessage 的MESSAGE_POST_RESULT 对这条消息的处理:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21private static class InternalHandler extends Handler {
       public InternalHandler(Looper looper) {
           super(looper);
       }
       @SuppressWarnings({"unchecked", "RawUseOfParameterizedType"})
       @Override
       public void handleMessage(Message msg) {
           AsyncTaskResult<?> result = (AsyncTaskResult<?>) msg.obj;
           switch (msg.what) {
               case MESSAGE_POST_RESULT:
                 //是他是他 就是他
                   // There is only one result
                   result.mTask.finish(result.mData[0]);
                   break;
               case MESSAGE_POST_PROGRESS:
                   result.mTask.onProgressUpdate(result.mData);
                   break;
           }
       }
   }
其中 result 是 AsyncTaskResult 类型,前面我们见到过的,mTask 当时我们传的是 this 也就是当前的 AsyncTask ,调用finish 方法,将mData 返回的结果传入进去,还记得我们前面看过的吗,将返回的结果存入AsyncTaskResult.mData中去了。
下面看下 finish方法:1
2
3
4
5
6
7
8
9
10
11private void finish(Result result) {
      //判断是否取消,如果取消了,就不执行onPostExecute 了
        if (isCancelled()) {
            onCancelled(result);
        } else {
          //4. 就执行onPostExecute 方法了
            onPostExecute(result);
        }
      // 将状态标志为 finish
        mStatus = Status.FINISHED;
    }
ok ,上述都是 mWorker 工作的,接下来是我们一开始说的 mFuture
- mFuture- 1 
 2
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18- private final FutureTask<Result> mFuture; 
 //初始化也是在AsyncTask 构造方法中执行的,在mWorker 之下,并且将mWorder 传入
 mFuture = new FutureTask<Result>(mWorker) {
 @Override
 protected void done() {
 try {
 postResultIfNotInvoked(get());
 } catch (InterruptedException e) {
 android.util.Log.w(LOG_TAG, e);
 } catch (ExecutionException e) {
 throw new RuntimeException("An error occurred while executing doInBackground()",
 e.getCause());
 } catch (CancellationException e) {
 postResultIfNotInvoked(null);
 }
 }
 };
postResultIfNotInvoked(get()); 如下:1
2
3
4
5
6
7
8
9
10
11private void postResultIfNotInvoked(Result result) {
        final boolean wasTaskInvoked = mTaskInvoked.get();
      //wasTaskInvoked 为true ,之前在 mWorker 中设置了为true
//mWorker = new WorkerRunnable<Params, Result>() {
//            public Result call() throws Exception {
 //               mTaskInvoked.set(true);
        if (!wasTaskInvoked) {
            postResult(result);
        }
    }
FutureTask :1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42我们知道mWorker implement Callable 接口,传入赋值给了callable 变量
 public FutureTask(Callable<V> callable) {
        if (callable == null)
            throw new NullPointerException();
        this.callable = callable;
        this.state = NEW;       // ensure visibility of callable
    }
public void run() {
        if (state != NEW ||
            !U.compareAndSwapObject(this, RUNNER, null, Thread.currentThread()))
            return;
        try {
          //callable 变量又赋值给了 c
            Callable<V> c = callable;
            if (c != null && state == NEW) {
                V result;
                boolean ran;
                try {
                  //这里调用c.call 实际上就是调用 mWorker.call 方法
//,由我们上面的分析知道,在mWorker.call 方法中最终会返回 result 结果
                    result = c.call();
                    ran = true;
                } catch (Throwable ex) {
                    result = null;
                    ran = false;
                    setException(ex);
                }
                if (ran)
                    set(result);
            }
        } finally {
            // runner must be non-null until state is settled to
            // prevent concurrent calls to run()
            runner = null;
            // state must be re-read after nulling runner to prevent
            // leaked interrupts
            int s = state;
            if (s >= INTERRUPTING)
                handlePossibleCancellationInterrupt(s);
        }
    }
ok ,这是 mFuture,还剩下最后一个:
exec.execute(mFuture);
exec 就是 sDefaultExecutor  ,其实 就是 SerialExecutor,如下:1
2
3private static volatile Executor sDefaultExecutor = SERIAL_EXECUTOR;
public static final Executor SERIAL_EXECUTOR = new SerialExecutor();
SerialExecutor 如下:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27private static class SerialExecutor implements Executor {
       final ArrayDeque<Runnable> mTasks = new ArrayDeque<Runnable>();
       Runnable mActive;
       public synchronized void execute(final Runnable r) {
           mTasks.offer(new Runnable() {
               public void run() {
                   try {
                       r.run();
                   } finally {
                       scheduleNext();
                   }
               }
           });
         //第一次肯定为null ,执行 scheduleNext
           if (mActive == null) {
               scheduleNext();
           }
       }
       protected synchronized void scheduleNext() {
         //给 mActivie 赋值,mTasks.poll 会从第一个开始取
           if ((mActive = mTasks.poll()) != null) {
               THREAD_POOL_EXECUTOR.execute(mActive);
           }
       }
   }
上面我们将 mFuture 传入,实际就是 r.mTask 是 ArrayDeque<Runnable> 姑且认为它是这个排队序列的吧。看下offer 方法:1
2
3
4
5
6
7
8
9
10
11
12
13/**
      插入一个 element 在队尾
     * Inserts the specified element at the end of this deque.
     *
     * <p>This method is equivalent to {@link #offerLast}.
     *
     * @param e the element to add
     * @return {@code true} (as specified by {@link Queue#offer})
     * @throws NullPointerException if the specified element is null
     */
    public boolean offer(E e) {
        return offerLast(e);
    }
看注释,也就是说是每次执行一个任务,都是在当前 deque 的队尾开始排队的。并且执行是串行的,因为当第二个线程过来的时候,判断 mActive 不为 null 将不会执行 scheduleNext.(我这个是8.0)源码,其实在 android 3.0 之后 AsyncTask 都是采用串行执行任务的。
各个版本的不同如下:android 1.6之前 ——  串行android 1.6-3.0 之间 —– 并行android 3.0 之后 —– 串行
尽管如此,我们仍然可以通过 调用 executeOnExecutor 来并行执行任务。
ok , 回到那个 execute 方法中,我们说调用了 r.run 实际山就是 调用 mFuture.run 方法:
上面我们展示过在 mFuture.run 方法中如下:
| 1 | try { | 
最终调用mWorker.call  方法,而在 mWorker.call 方法中,我们完成一系列的任务,调用了 doInBackground onPostExecute 完成了整个的调用过程。
有的人可能已经注意到了 还差一个 onProgressUpdate 方法还没被调用,我们知道只有调用那个 publishProgress 方法的时候才能调用 onProgressUpdate ,那下面我们卡夏 publishProgress 方法:1
2
3
4
5
6
7
8@WorkerThread
    protected final void publishProgress(Progress... values) {
        //如果没取消
        if (!isCancelled()) {
    //会发送一个 MESSAGE_POST_PROGRESS 的消息            getHandler().obtainMessage(MESSAGE_POST_PROGRESS,
                    new AsyncTaskResult<Progress>(this, values)).sendToTarget();
        }
    }
| 1 | private static class InternalHandler extends Handler { | 
会调用 AsyncTask 的 onProgressUpdate 方法了。结束。
总结
AsyncTask实现原理就内部维护了2个线程池+Handlder,2个线程池分别为:任务队列 线程池(SerialExecutor),执行 线程池(Thread_pool_executor)真正执行具体的线程任务,任务完成最后通过InternalHandler发送消息 ,将执行任务的结果post到主线程中。此线程池中执行任务的顺序,根据不同api版本的不同顺序略有不同,在android 1.6之前,是串行执行任务,在Android 1.6-3.0之间通过并行执行任务,Android 3.0之后就是默认串行执行任务了,当然我们也可以通过调用executeOnExecuteor并行执行。