AsyncTask 源碼閱讀筆記

* <p>AsyncTask is designed to be a helper class around {@link Thread} and {@link Handler}
* and does not constitute a generic threading framework. AsyncTasks should ideally be
* used for short operations (a few seconds at the most.) If you need to keep threads
* running for long periods of time, it is highly recommended you use the various APIs
* provided by the <code>java.util.concurrent</code> package such as {@link Executor},
* {@link ThreadPoolExecutor} and {@link FutureTask}.</p>

AsyncTask 應該被用來處理短期(幾秒)的操做。若是想要讓線程長時間的運行,強烈推薦使用 java.util.concurrent 包下面的工具。

  

 

* <p>AsyncTask must be subclassed to be used. The subclass will override at least
* one method ({@link #doInBackground}), and most often will override a
* second one ({@link #onPostExecute}.)</p>

AsyncTask 是抽象類,子類須要實現其抽象方法,實例化後才能使用,下面是一個使用例子:
* <p>Here is an example of subclassing:</p>
* <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");
*     }
* }

  

 
* <h2>AsyncTask's generic types</h2>
* <p>The three types used by an asynchronous task are the following:</p>
* <ol>
*     <li><code>Params</code>, the type of the parameters sent to the task upon
*     execution.</li>
*     <li><code>Progress</code>, the type of the progress units published during
*     the background computation.</li>
*     <li><code>Result</code>, the type of the result of the background
*     computation.</li>
* </ol>
* <p>Not all types are always used by an asynchronous task. To mark a type as unused,
* simply use the type {@link Void}:</p>
* <pre>
* private class MyTask extends AsyncTask<Void, Void, Void> { ... }
* </pre>

AsyncTask 的三個泛型參數:
Params --- Task 執行時須要的參數類型
Progress --- Task 執行時的進行類型
Result --- Task 執行完成以後返回的結果類型

  

 
* <h2>The 4 steps</h2>
* <p>When an asynchronous task is executed, the task goes through 4 steps:</p>
* <ol>
*     <li>{@link #onPreExecute()}, invoked on the UI thread before the task
*     is executed. This step is normally used to setup the task, for instance by
*     showing a progress bar in the user interface.</li>
*     <li>{@link #doInBackground}, invoked on the background thread
*     immediately after {@link #onPreExecute()} finishes executing. This step is used
*     to perform background computation that can take a long time. The parameters
*     of the asynchronous task are passed to this step. The result of the computation must
*     be returned by this step and will be passed back to the last step. This step
*     can also use {@link #publishProgress} to publish one or more units
*     of progress. These values are published on the UI thread, in the
*     {@link #onProgressUpdate} step.</li>
*     <li>{@link #onProgressUpdate}, invoked on the UI thread after a
*     call to {@link #publishProgress}. The timing of the execution is
*     undefined. This method is used to display any form of progress in the user
*     interface while the background computation is still executing. For instance,
*     it can be used to animate a progress bar or show logs in a text field.</li>
*     <li>{@link #onPostExecute}, invoked on the UI thread after the background
*     computation finishes. The result of the background computation is passed to
*     this step as a parameter.</li>
* </ol>

AsyncTask 執行的時候,下面四個方法將會順序執行:
onPreExecute() 在主線程執行,在Task執行前調用。
doInBackground() 在後臺線程執行,onPreExecute() 執行完以後,該方法當即執行。
onProgressUpdate() 在主線程執行,可用來展現 Task 的進度。
onPostExecute() 在主線程執行,Task 執行完以後,該方法執行。

  

 
* <h2>Cancelling a task</h2>
* <p>A task can be cancelled at any time by invoking {@link #cancel(boolean)}. Invoking
* this method will cause subsequent calls to {@link #isCancelled()} to return true.
* After invoking this method, {@link #onCancelled(Object)}, instead of
* {@link #onPostExecute(Object)} will be invoked after {@link #doInBackground(Object[])}
* returns. To ensure that a task is cancelled as quickly as possible, you should always
* check the return value of {@link #isCancelled()} periodically from
* {@link #doInBackground(Object[])}, if possible (inside a loop for instance.)</p>

取消任務,調用 cancel() 方法,調用該方法以後 isCancel() 方法會返回 true。
而且,onPostExecute() 方法不會執行,onCancelled() 會執行。
爲了保證 Task 儘快取消,你應該在 doInBackground() 中隨時檢查 isCancelled() 的返回值。

  

AsyncTask 的構造函數java

    /**
     * Creates a new asynchronous task. This constructor must be invoked on the UI thread.
     */
    public AsyncTask() {
        mWorker = new WorkerRunnable<Params, Result>() {
            public Result call() throws Exception {
                mTaskInvoked.set(true);

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

        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 occured while executing doInBackground()",
                            e.getCause());
                } catch (CancellationException e) {
                    postResultIfNotInvoked(null);
                }
            }
        };
    }

在構造函數中,實例化了兩個變量android

mWorker 就是一個 Callable 的實現類。多線程

mFuture 須要看一下 FutureTask 的源碼。async

 

下面看最主要的方法:ide

    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;
    }

executeOnExecutor 方法能夠指定 Executor 來執行多線程服務。工具

當咱們調用了 execute 方法的時候,從代碼中能夠看出 onPreExecute() 會先執行。oop

而後是 exec.execute(mFuture) 執行,因爲 mFuture 也實現了 Runnable 對象,而且還實現了 Callable 對象。post

即 mFuture 的 run() 方法被調用:ui

public void run() {
        if (state != NEW ||
            !UNSAFE.compareAndSwapObject(this, runnerOffset,
                                         null, Thread.currentThread()))
            return;
        try {
            Callable<V> c = callable;
            if (c != null && state == NEW) {
                V result;
                boolean ran;
                try {
                    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);
        }
    }

其中,調用了 callable 的 call() 方法。由構造函數可知,mWorker 的 call() 方法被調用。

即執行了 doInBackground() 方法。而後,run() 方法的最後,會調用到 mFuture 的 done() 方法(省略一連串的調用)。

即 finish() 方法被調用(省略一連串的調用)。

    private void finish(Result result) {
        if (isCancelled()) {
            onCancelled(result);
        } else {
            onPostExecute(result);
        }
        mStatus = Status.FINISHED;
    }

能夠看出,onPostExecute() 會被調用.....


onProgressUpdate() 由下面的方法觸發:
    /**
     * This method can be invoked from {@link #doInBackground} to
     * publish updates on the UI thread while the background computation is
     * still running. Each call to this method will trigger the execution of
     * {@link #onProgressUpdate} on the UI thread.
     *
     * {@link #onProgressUpdate} will not be called if the task has been
     * canceled.
     *
     * @param values The progress values to update the UI with.
     *
     * @see #onProgressUpdate
     * @see #doInBackground
     */
    protected final void publishProgress(Progress... values) {
        if (!isCancelled()) {
            sHandler.obtainMessage(MESSAGE_POST_PROGRESS,
                    new AsyncTaskResult<Progress>(this, values)).sendToTarget();
        }
    }

  

PS:序列執行器

    private 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();
                    }
                }
            });
            if (mActive == null) {
                scheduleNext();
            }
        }

        protected synchronized void scheduleNext() {
            if ((mActive = mTasks.poll()) != null) {
                THREAD_POOL_EXECUTOR.execute(mActive);
            }
        }
    }
相關文章
相關標籤/搜索