java.util.concurrent包(4)——Callable和Future

Callable和Future,一個產生結果,一個拿到結果。編程


Callable接口相似於Runnable,從名字就能夠看出來了,可是Runnable不會返回結果,而且沒法拋出返回結果的異常,而Callable功能更強大一些,被線程執行後,能夠返回值,這個返回值能夠被Future拿到,也就是說,Future能夠拿到異步執行任務的返回值,下面來看一個簡單的例子:併發


public class CallableAndFuturedom

{異步

public static void main(String[] args)ide

{spa

Callable<Integer> callable = new Callable<Integer>().net

{線程

public Integer call() throws Exceptionhtm

{對象

return new Random().nextInt(100);

}

};

FutureTask<Integer> future = new FutureTask<Integer>(callable);

new Thread(future).start();

try

{

// 作其餘的事情

Thread.sleep(5000);

System.out.println(future.get());

}

catch (InterruptedException e)

{

e.printStackTrace();

}

catch (ExecutionException e)

{

e.printStackTrace();

}

}

}

FutureTask實現了兩個接口,Runnable和Future,因此既能夠做爲Runnable被線程執行,又可做爲Future獲得Callable的返回值,這個組合的使用有什麼好處呢?假設有一個很耗時的返回值須要計算,而且這個返回值不是馬上須要的話,就可使用這個組合,用另外一個線程去計算返回值,而當前線程在使用這個返回值以前能夠作其它的操做,等到須要這個返回值時,再經過Future獲得,豈不美哉!這裏有一個Future模式的介紹:http://openhome.cc/Gossip/DesignPattern/FuturePattern.htm。


下面來看另外一種方式使用Callable和Future,經過ExecutorService的submit方法執行Callable,並返回Future。


public class CallableAndFuture2

{

public static void main(String[] args)

{

ExecutorService threadpool = Executors.newFixedThreadPool(1);

Future<String> future = threadpool.submit(new Callable<String>()

{

public String call() throws Exception

{

return "XY";

}

});


try

{

// 作其餘的事情

Thread.sleep(5000);

System.out.println(future.get());

}

catch (InterruptedException e)

{

e.printStackTrace();

}

catch (ExecutionException e)

{

e.printStackTrace();

}

}

}

代碼是否是簡化了不少,ExecutorService繼承自Executor,它的目的是爲管理Thread對象,從而簡化併發編程,Executor使我無需顯示的去管理線程的生命週期,是JDK 5以後啓動任務的首選方式。


執行多個帶返回值的任務,並取得多個返回值,代碼以下:

public class CallableAndFuture3

{

public static void main(String[] args)

{

ExecutorService threadPool = Executors.newCachedThreadPool();

CompletionService<Integer> cs 

                             = new ExecutorCompletionService<Integer>(threadPool);

for (int i = 1; i <= 10; i++)

{

final int taskID = i;

cs.submit(new Callable<Integer>()

{

public Integer call() throws Exception

{

return taskID;

}

});

}

// 作其餘的事情

for (int i = 1; i <= 10; i++)

{

try

{

System.out.println(cs.take().get());

}

catch (InterruptedException e)

{

e.printStackTrace();

}

catch (ExecutionException e)

{

e.printStackTrace();

}

}

}

}


原帖地址:http://blog.csdn.net/ghsau/article/details/7451464

相關文章
相關標籤/搜索