Javaの並行性-先物と呼び出し可能
java.util.concurrent.Callableオブジェクトは、スレッドのみを実行できる実行可能なインターフェースとは対照的に、スレッドによって実行された計算結果を返すことができます。Callableオブジェクトは、スレッドによって実行されているタスクの進行状況を監視するメソッドを提供するFutureオブジェクトを返します。Futureオブジェクトを使用して、Callableのステータスを確認し、スレッドが完了したらCallableから結果を取得できます。また、タイムアウト機能も提供します。
構文
//submit the callable using ThreadExecutor
//and get the result as a Future object
Future<Long> result10 = executor.submit(new FactorialService(10));
//get the result using get method of the Future object
//get method waits till the thread execution and then return the result of the execution.
Long factorial10 = result10.get();
例
次のTestThreadプログラムは、スレッドベースの環境でのFuturesとCallablesの使用法を示しています。
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
public class TestThread {
public static void main(final String[] arguments) throws InterruptedException,
ExecutionException {
ExecutorService executor = Executors.newSingleThreadExecutor();
System.out.println("Factorial Service called for 10!");
Future<Long> result10 = executor.submit(new FactorialService(10));
System.out.println("Factorial Service called for 20!");
Future<Long> result20 = executor.submit(new FactorialService(20));
Long factorial10 = result10.get();
System.out.println("10! = " + factorial10);
Long factorial20 = result20.get();
System.out.println("20! = " + factorial20);
executor.shutdown();
}
static class FactorialService implements Callable<Long> {
private int number;
public FactorialService(int number) {
this.number = number;
}
@Override
public Long call() throws Exception {
return factorial();
}
private Long factorial() throws InterruptedException {
long result = 1;
while (number != 0) {
result = number * result;
number--;
Thread.sleep(100);
}
return result;
}
}
}
これにより、次の結果が得られます。
出力
Factorial Service called for 10!
Factorial Service called for 20!
10! = 3628800
20! = 2432902008176640000