Since use ExecutorService
can submit
a Callable
task and return a Future
, why need to use FutureTask
to wrap Callable
task and use the method execute
? I feel they both do the same thing.
What's the difference between Future and FutureTask in Java?
34.6k views Asked by Hesey AtThere are 6 answers
Future
is just interface. Behind the scene, the implementation is FutureTask
.
You can absolutely use FutureTask
manually but you will lose the advantages of using Executor
(pooling thread, limit the thread, etc). Using FutureTask
is quite similar to using the old Thread
and using the run method.
FutureTask
This class provides a base implementation of Future
, with methods to start and cancel a computation
Future is the interface
As already mentioned , but speaking not in general but in more of technical terms, since FutureTask implements RunnableFuture, you can call it using
FutureTask<T> result = new FutureTask<T>(new #YourClassImplementingCallable());
Thread t1= new Thread(result);
t1.start();
Object<T> obj = result.get();
This is more inline with older runnable but it also has the capacity to return result via callback.
The greater power of FutureTask over Future lies in the fact that it has more control over threads as against just simply Submitting a callable to Future and let executor handle the threads.
like you can call here t1.join().
As Mark, and others, correctly answered that Future
is the interface for FutureTask
and Executor
effectively its factory; meaning that application code rarely instantiates FutureTask
directly. To complement the discussion I am providing an example showing a situation where FutureTask
is constructed and used directly, outside any Executor
:
FutureTask<Integer> task = new FutureTask<Integer>(()-> {
System.out.println("Pretend that something complicated is computed");
Thread.sleep(1000);
return 42;
});
Thread t1 = new Thread(()->{
try {
int r = task.get();
System.out.println("Result is " + r);
} catch (InterruptedException | ExecutionException e) {}
});
Thread t2 = new Thread(()->{
try {
int r = task.get();
System.out.println("Result is " + r);
} catch (InterruptedException | ExecutionException e) {}
});
Thread t3 = new Thread(()->{
try {
int r = task.get();
System.out.println("Result is " + r);
} catch (InterruptedException | ExecutionException e) {}
});
System.out.println("Several threads are going to wait until computations is ready");
t1.start();
t2.start();
t3.start();
task.run(); // let the main thread to compute the value
Here, FutureTask
is used as a synchronization tool, like CountdownLatch
or similar barrier primitive. It could have been re-implemented by using CountdownLatch
or locks and conditions; FutureTask
just makes it nicely encapsulated, self-explanatory, elegant and with less code.
Also note that FutureTask#run() method must be called explicitly, in any of the threads; there no Executor around to do it for you. In my code, it is eventually executed by the main thread, but one can modify get()
method to call run()
on the first thread calling get()
, therefore the first thread reaching get()
, and it can be any of T1, T2 or T3, would do the calculation for all remaining threads.
On this idea - first thread requesting result would do the calculation for others, while concurrent attempts would be blocked - is based Memoizer, see Memoizer Cache example from page 108 in "Java Concurrency in Practice".
In fact you are correct. The two approaches are identical. You generally don't need to wrap them yourself. If you are, you're likely duplicating the code in AbstractExecutorService:
The only difference between Future and RunnableFuture, is the run() method:
A good reason to let the Executor construct the FutureTask for you is to ensure that there is no possible way more than one reference exists to the FutureTask instance. That is, the Executor owns this instance.