本節(jié)展示如何在線程池里執(zhí)行任務(wù)。流程是,添加一個任務(wù)到線程池的工作隊列,當有線程可用時(執(zhí)行完其他任務(wù),空閑,或者還沒執(zhí)行任務(wù)),ThreadPoolExecutor會從隊列里取任務(wù),并在線程里運行。
本課同時向你展示了如何停止正在運行的任務(wù)。
在線程池里的線程上執(zhí)行任務(wù)
在ThreadPoolExecutor.execute()里傳入 Runnable對象啟動任務(wù)。這個方法會把任務(wù)添加到線程池工作隊列。當有空閑線程時,管理器會取出等待最久的任務(wù),在線程上運行。
public class PhotoManager {
    public void handleState(PhotoTask photoTask, int state) {
        switch (state) {
            // The task finished downloading the image
            case DOWNLOAD_COMPLETE:
            // Decodes the image
                mDecodeThreadPool.execute(
                        photoTask.getPhotoDecodeRunnable());
            ...
        }
        ...
    }
    ...
}
當ThreadPoolExecutor啟動Runnable時,會自動調(diào)用run()方法。
中斷正在運行的代碼
要停止任務(wù),你需要中斷任務(wù)的進程。你需要在創(chuàng)建任務(wù)的時候,保存一個當前線程的handle.
如:
class PhotoDecodeRunnable implements Runnable {
    // Defines the code to run for this task
    public void run() {
        /*
         * Stores the current Thread in the
         * object that contains PhotoDecodeRunnable
         */
        mPhotoTask.setImageDecodeThread(Thread.currentThread());
        ...
    }
    ...
}
要中斷線程,調(diào)用Thread.interrupt()就可以了。提示:線程對象是系統(tǒng)控制的,可以在你的app進程外被編輯。因為這個原因,你需要在中斷它前加訪問鎖,放到一個同步塊里:
public class PhotoManager {
    public static void cancelAll() {
        /*
         * Creates an array of Runnables that's the same size as the
         * thread pool work queue
         */
        Runnable[] runnableArray = new Runnable[mDecodeWorkQueue.size()];
        // Populates the array with the Runnables in the queue
        mDecodeWorkQueue.toArray(runnableArray);
        // Stores the array length in order to iterate over the array
        int len = runnableArray.length;
        /*
         * Iterates over the array of Runnables and interrupts each one's Thread.
         */
        synchronized (sInstance) {
            // Iterates over the array of tasks
            for (int runnableIndex = 0; runnableIndex < len; runnableIndex++) {
                // Gets the current thread
                Thread thread = runnableArray[taskArrayIndex].mThread;
                // if the Thread exists, post an interrupt to it
                if (null != thread) {
                    thread.interrupt();
                }
            }
        }
    }
    ...
}
在大多數(shù)案例里,Thread.interrupt()會馬上停止線程。可是,它只會停止在等待的線程,但不會中斷cpu或network-intensive任務(wù)。為了避免系統(tǒng)變慢,你應(yīng)該在開始嘗試操作前測試等待中斷的請求。
/*
 * Before continuing, checks to see that the Thread hasn't
 * been interrupted
 */
if (Thread.interrupted()) {
    return;
}
...
// Decodes a byte array into a Bitmap (CPU-intensive)
BitmapFactory.decodeByteArray(
        imageBuffer, 0, imageBuffer.length, bitmapOptions);
...