import java.util.concurrent.Callable;
import java.util.concurrent.CompletionService;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorCompletionService;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
public class TestCompletionService {
public static void main(String[] args) throws InterruptedException,
ExecutionException {
ExecutorService exec = Executors.newFixedThreadPool(10);
CompletionService serv = new ExecutorCompletionService(exec);
for (int index = 0; index < 5; index++) {
final int NO = index;
Callable downImg = new Callable() {
public String call() throws Exception {
Thread.sleep((long) (Math.random() * 10000));
return "Downloaded Image " + NO;
}
};
serv.submit(downImg);
}
Thread.sleep(1000 * 2);
System.out.println("Show web content");
for (int index = 0; index < 5; index++) {
Future task = serv.take();
String img = (String)task.get();
System.out.println(img);
}
System.out.println("End");
// 關閉線程池
exec.shutdown();
}
}
考慮以下情境:瀏覽網頁時,瀏覽器了5個線程下載網頁中的圖片檔案,由於圖片大小、網站訪問速度等諸多因素的影響,完成圖片下載的時間就會有很大的不同。如果先下載完成的圖片就會被先顯示到介面上,反之,後下載的圖片就後顯示。
Java的並發庫的CompletionService可
以滿足這種情境要求。該介面有兩個重要方法:submit()和take()。submit用於提交一個runnable或者callable,一般會提
交給一個線程池處理;而take就是取出已經執行完畢runnable或者callable執行個體的Future對象,如果沒有滿足要求的,就等待了。 CompletionService還有一個對應的方法poll,該方法與take類似,只是不會等待,如果沒有滿足要求,就返回null對象。