CompletableFuture
Doug Lea
dl at cs.oswego.edu
Thu Nov 29 04:22:08 PST 2012
On 11/28/12 15:34, Doug Lea wrote:
> Here's how to make trees/dags:
>
> t1 = new CompletableFuture();
> t2 = t1.then(fun1);
> t3 = t1.then(fun2);
> t4 = t2.then(fun3);
> t5 = new CompletableFuture();
> t6 = t1.and(t3, fun4);
(typo; should be t6 = t5.and(t3, fun4);
>
> This gives you (hoping the ascii-arg makes it through):
>
>
> / -> t2 -> t4
> t1
> \ -> t3
> \
> t5 -----> & t6
>
> Does that help?
>
Also, here's an update that contains a start on fleshing out some
of the mechanics and intent, and adds a few things based on
my own mangling of some suggestions by Chris Poviik (Google).
/*
* Written by Doug Lea with assistance from members of JCP JSR-166
* Expert Group and released to the public domain, as explained at
* http://creativecommons.org/publicdomain/zero/1.0/
*/
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.TimeoutException;
/**
* A {@link Future} that may be explicitly completed (thus setting its
* value and status), and may support dependent CompletableFutures
* that trigger upon its completion.
*
* <p>Functions and actions supplied for dependent completions using
* {@code then}, {@code and}, {@code or}, and {@code exceptionally}
* are performed by the thread that completes the current
* CompletableFuture, or by the caller of the method, if already
* complete.
*
* <p> When two or more threads attempt to {@link #complete} or {@link
* #completeExceptionally} a CompletableFuture, only one of them will
* succeed. When completion entails computation of a function (or void
* action), the function is executed <em>after</em> establishing
* precedence. If this function terminates abruptly with an exception,
* then method {@code complete} acts as {@code completeExceptionally}
* with that exception. Method {@link #isCompleting} can be used to
* determine if a completion function gets "stuck", and method {@link
* #force} can be used in attempts to recover.
*
* <p>This class itself does not provide explicit support for
* asynchronous execution, but can be used in conjunction with several
* forms of concurrency. For examples, a new CompletableFuture may be
* constructed with (a function of) the results of an asynchronous
* task. Also, a supplied dependent action may itself fork a new task.
*
*/
public class CompletableFuture<T> implements Future<T> {
/**
* Creates a new incomplete CompletableFuture.
*/
public CompletableFuture();
/**
* Creates a new CompletableFuture that is completed with the
* result of the given Future, if/when it completes.
*/
public CompletableFuture(Future<T> f);
/**
* Creates a new CompletableFuture that is completed with the
* result of applying the given function to the result of the
* given Future, if/when it completes.
*/
public <U> CompletableFuture(Future<U> f, Function<? super U, T> fn);
/**
* If not already completed, sets the value returned by get() and
* related methods to the given value.
*
* @return true if this invocation of {@code complete} caused
* this CompletableFuture to transition to a completed state,
* else false.
*/
public boolean complete(T value);
/**
* If not already completed, causes invocations of get() to
* throw the given exception.
*
* @return true if this invocation of {@code
* completeExceptionally} caused this CompletableFuture to
* transition to a completed state, else false.
*/
public boolean completeExceptionally(Throwable ex);
// Cascading functional completions
/**
* Creates and returns a CompletableFuture that is completed with
* the result of the given function of this CompletableFuture's
* result if/when this completes normally.
*
* @return the new CompletableFuture
*/
public <U> CompletableFuture<U> then(Function<? super T,? extends U> fn);
/**
* Creates and returns a CompletableFuture that is completed with
* the result of the given function of this and the other given
* CompletableFuture's results if/when both complete normally.
*
* @return the new CompletableFuture
*/
public <U,V> CompletableFuture<V> and(CompletableFuture<? extends U> x,
BiFunction<? super T,? super U,?
extends V> fn);
/**
* Creates and returns a CompletableFuture that is completed with
* the result of the given function of either this or the other
* given CompletableFuture's results if/when either complete
* normally.
*
* @return the new CompletableFuture
*/
public <U> CompletableFuture<U> or(CompletableFuture<? extends T> x,
BiFunction<? super T,? super T, ?
extends U> fn);
// Action (vs function) forms
/**
* Creates and returns a CompletableFuture that is completed
* after performing the given action with the result of this
* CompletableFuture if/when it completes normally.
*
* @return the new CompletableFuture
*/
public CompletableFuture<Void> then(Block<? super T> action);
/**
* Creates and returns a CompletableFuture that is completed
* after performing the given action with the result of the given
* function of this and the other given CompletableFuture's
* results if/when both complete normally.
*
* @return the new CompletableFuture
*/
public <U> CompletableFuture<Void> and(CompletableFuture<? extends U> x,
BiBlock<? super T, ? super U> action);
/**
* Creates and returns a CompletableFuture that is completed
* after performing the given action with the result of either
* this or the other given CompletableFuture's results if/when
* either complete normally.
*
* @return the new CompletableFuture
*/
public CompletableFuture<Void> or(CompletableFuture<? extends T> x,
BiBlock<? super T, ? super T> action);
/**
* Creates and returns a CompletableFuture that is completed after
* performing the given action with the exception triggering this
* CompletableFuture's completion if/when it completes
* exceptionally.
*
* @return the new CompletableFuture
*/
public CompletableFuture<Void> exceptionally(Block<Throwable> action);
// extracting results
/**
* Returns the value when complete, or throws an (unchecked)
* exception if completed exceptionally. (The checked exception
* variant is available using the timed form of get.)
*/
public T get();
/**
* Returns the value if completed, else the given valueIfAbsent.
*/
public T getNow(T valueIfAbsent);
/**
* Waits if necessary for at most the given time for the computation
* to complete, and then retrieves its result, if available.
*
* @param timeout the maximum time to wait
* @param unit the time unit of the timeout argument
* @return the computed result
* @throws CancellationException if the computation was cancelled
* @throws ExecutionException if the computation threw an
* exception
* @throws InterruptedException if the current thread was interrupted
* while waiting
* @throws TimeoutException if the wait timed out
*/
T get(long timeout, TimeUnit unit)
throws InterruptedException, ExecutionException, TimeoutException;
// misc
/**
* Whether or not already completed, sets the value subsequently
* returned by method get() and related methods to the given
* value. This method is designed primarily for use in recovery
* actions.
*
* @param value the completion value
*/
public void force(T value);
/**
* Returns true if this CompletableFuture is in the process of
* completing but has not yet completed. This method is designed
* to help diagnose cases in which supplied completion functions
* become "stuck" in the process of completion.
*/
public boolean isCompleting();
/**
* Attempts to cancel further processing of this task. This
* attempt will fail if already completed. If successful,
* subsequent calls to {@link #isCancelled}, {@link #isDone}, and
* {@code cancel} will return {@code true} and calls to {@link
* #get} and related methods will result in {@code
* CancellationException}.
*
* @param mayInterruptIfRunning this value has no effect in the
* default implementation because interrupts are not used to
* control processing.
*
* @return {@code true} if this task is now cancelled
*/
public boolean cancel(boolean mayInterruptIfRunning);
/**
* Returns {@code true} if this CompletableFuture was cancelled
* before it completed normally.
*
* @return {@code true} if this CompletableFuture was cancelled
* before it completed normally
*/
public boolean isCancelled();
/**
* Returns {@code true} if completed in any fashion: normally,
* exceptionally, or cancellation.
*
* @return {@code true} if completed
*/
public boolean isDone();
}
More information about the lambda-libs-spec-observers
mailing list