編寫:AllenZheng1991 - 原文:http://developer.android.com/training/multiple-threads/create-threadpool.html
在前面的課程中展示瞭如何在單獨的一個線程中執行一個任務。如果你的線程只想執行一次,那麼上一課的內容已經能滿足你的需要了。
如果你想在一個數據集中重複執行一個任務,而且你只需要一個執行運行一次。這時,使用一個IntentService將能滿足你的需求。 為了在資源可用的的時候自動執行任務,或者允許不同的任務同時執行(或前後兩者),你需要提供一個管理線程的集合。 為了做這個管理線程的集合,使用一個ThreadPoolExecutor實例,當一個線程在它的線程池中變得不受約束時,它會運行隊列中的一個任務。 為了能執行這個任務,你所需要做的就是把它加入到這個隊列。
一個線程池能運行多個並行的任務實例,因此你要能保證你的代碼是線程安全的,從而你需要給會被多個線程訪問的變量附上同步代碼塊(synchronized block)。 當一個線程在對一個變量進行寫操作時,通過這個方法將能阻止另一個線程對該變量進行讀取操作。 典型的,這種情況會發生在靜態變量上,但同樣它也能突然發生在任意一個只實例化一次。 為了學到更多的相關知識,你可以閱讀進程與線程這一API指南。
在自己的類中實例化ThreadPoolExecutor類。在這個類裡需要做以下事:
1. 為線程池使用靜態變量
為了有一個單一控制點用來限制CPU或涉及網絡資源的Runnable類型,你可能需要有一個能管理所有線程的線程池,且每個線程都會是單個實例。比如,你可以把這個作為一部分添加到你的全局變量的聲明中去:
public class PhotoManager {
...
static {
...
// Creates a single static instance of PhotoManager
sInstance = new PhotoManager();
}
...
2. 使用私有構造方法
讓構造方法私有從而保證這是一個單例,這意味著你不需要在同步代碼塊(synchronized block)中額外訪問這個類:
public class PhotoManager {
...
/**
* Constructs the work queues and thread pools used to download
* and decode images. Because the constructor is marked private,
* it's unavailable to other classes, even in the same package.
*/
private PhotoManager() {
...
}
3.通過調用線程池類裡的方法開啟你的任務
在線程池類中定義一個能添加任務到線程池隊列的方法。例如:
public class PhotoManager {
...
// Called by the PhotoView to get a photo
static public PhotoTask startDownload(
PhotoView imageView,
boolean cacheFlag) {
...
// Adds a download task to the thread pool for execution
sInstance.
mDownloadThreadPool.
execute(downloadTask.getHTTPDownloadRunnable());
...
}
4. 在構造方法中實例化一個Handler,且將它附加到你APP的UI線程。
一個Handler允許你的APP安全地調用UI對象(例如 View對象)的方法。大多數UI對象只能從UI線程安全的代碼中被修改。這個方法將會在與UI線程進行通信(Communicate with the UI Thread)這一課中進行詳細的描述。例如:
private PhotoManager() {
...
// Defines a Handler object that's attached to the UI thread
mHandler = new Handler(Looper.getMainLooper()) {
/*
* handleMessage() defines the operations to perform when
* the Handler receives a new Message to process.
*/
@Override
public void handleMessage(Message inputMessage) {
...
}
...
}
}
一旦有了整體的類結構,你可以開始定義線程池了。為了初始化一個ThreadPoolExecutor對象,你需要提供以下數值:
1. 線程池的初始化大小和最大的大小
這個是指最初分配給線程池的線程數量,以及線程池中允許的最大線程數量。在線程池中擁有的線程數量主要取決於你的設備的CPU內核數。
這個數字可以從系統環境中獲得:
public class PhotoManager {
...
/*
* Gets the number of available cores
* (not always the same as the maximum number of cores)
*/
private static int NUMBER_OF_CORES =
Runtime.getRuntime().availableProcessors();
}
這個數字可能並不反映設備的物理核心數量,因為一些設備根據系統負載關閉了一個或多個CPU內核,對於這樣的設備,availableProcessors()
方法返回的是處於活動狀態的內核數量,可能少於設備的實際內核總數。
2.線程保持活動狀態的持續時間和時間單位
這個是指線程被關閉前保持空閒狀態的持續時間。這個持續時間通過時間單位值進行解譯,是TimeUnit()中定義的常量之一。
3.一個任務隊列
這個傳入的隊列由ThreadPoolExecutor獲取的Runnable對象組成。為了執行一個線程中的代碼,一個線程池管理者從先進先出的隊列中取出一個Runnable對象且把它附加到一個線程。當你創建線程池時需要提供一個隊列對象,這個隊列對象類必須實現BlockingQueue接口。為了滿足你的APP的需求,你可以選擇一個Android SDK中已經存在的隊列實現類。為了學習更多相關的知識,你可以看一下ThreadPoolExecutor類的概述。下面是一個使用LinkedBlockingQueue實現的例子:
public class PhotoManager {
...
private PhotoManager() {
...
// A queue of Runnables
private final BlockingQueue<Runnable> mDecodeWorkQueue;
...
// Instantiates the queue of Runnables as a LinkedBlockingQueue
mDecodeWorkQueue = new LinkedBlockingQueue<Runnable>();
...
}
...
}
##創建一個線程池
為了創建一個線程池,可以通過調用ThreadPoolExecutor()構造方法初始化一個線程池管理者對象,這樣就能創建和管理一組可約束的線程了。如果線程池的初始化大小和最大大小相同,ThreadPoolExecutor在實例化的時候就會創建所有的線程對象。例如:
private PhotoManager() {
...
// Sets the amount of time an idle thread waits before terminating
private static final int KEEP_ALIVE_TIME = 1;
// Sets the Time Unit to seconds
private static final TimeUnit KEEP_ALIVE_TIME_UNIT = TimeUnit.SECONDS;
// Creates a thread pool manager
mDecodeThreadPool = new ThreadPoolExecutor(
NUMBER_OF_CORES, // Initial pool size
NUMBER_OF_CORES, // Max pool size
KEEP_ALIVE_TIME,
KEEP_ALIVE_TIME_UNIT,
mDecodeWorkQueue);
}