com.landawn.abacus.util.CompletableFuture Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of abacus-android Show documentation
Show all versions of abacus-android Show documentation
A general programming library in Java
/*
* Copyright (C) 2016 HaiYang Li
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except
* in compliance with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed under the License
* is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express
* or implied. See the License for the specific language governing permissions and limitations under
* the License.
*/
package com.landawn.abacus.util;
import java.util.Arrays;
import java.util.List;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Executor;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.FutureTask;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import com.landawn.abacus.logging.Logger;
import com.landawn.abacus.logging.LoggerFactory;
import com.landawn.abacus.util.Tuple.Tuple4;
import com.landawn.abacus.util.function.BiConsumer;
import com.landawn.abacus.util.function.BiFunction;
import com.landawn.abacus.util.function.Consumer;
import com.landawn.abacus.util.function.Function;
/**
*
* @since 0.8
*
* @author Haiyang Li
*
* @see https://docs.oracle.com/javase/8/docs/api/java/util/concurrent/CompletableFuture.html
*/
public class CompletableFuture implements Future {
private static final Logger logger = LoggerFactory.getLogger(CompletableFuture.class);
private static final ExecutorService commonPool = Executors.newFixedThreadPool(64);
static {
Runtime.getRuntime().addShutdownHook(new Thread() {
@Override
public void run() {
try {
commonPool.shutdown();
commonPool.awaitTermination(180, TimeUnit.SECONDS);
} catch (Throwable e) {
logger.error("Failed to commit the tasks in queue in ExecutorService before shutdown", e);
}
}
});
}
final Future future;
final List> upFutures;
final Executor asyncExecutor;
CompletableFuture(final Future future, final List> upFutures, final Executor asyncExecutor) {
this.future = future;
this.upFutures = upFutures;
this.asyncExecutor = asyncExecutor;
}
public static CompletableFuture run(final Runnable action) {
return run(action, commonPool);
}
public static CompletableFuture run(final Try.Callable action) {
return run(action, commonPool);
}
public static CompletableFuture run(final Runnable action, final Executor executor) {
final FutureTask futureTask = new FutureTask<>(action, null);
executor.execute(futureTask);
return new CompletableFuture<>(futureTask, null, executor);
}
public static CompletableFuture run(final Try.Callable action, final Executor executor) {
final FutureTask futureTask = new FutureTask<>(action);
executor.execute(futureTask);
return new CompletableFuture<>(futureTask, null, executor);
}
/**
*
* @param result
* @param asyncExecutor
* @return a CompletableFuture which is already done by passing the result to it directly.
*/
public static CompletableFuture completed(final T result) {
return new CompletableFuture<>(new Future() {
@Override
public boolean cancel(boolean mayInterruptIfRunning) {
return false;
}
@Override
public boolean isCancelled() {
return false;
}
@Override
public boolean isDone() {
return true;
}
@Override
public T get() {
return result;
}
@Override
public T get(final long timeout, final TimeUnit unit) {
return result;
}
}, null, commonPool);
}
@Override
public boolean cancel(boolean mayInterruptIfRunning) {
return future.cancel(mayInterruptIfRunning);
}
@Override
public boolean isCancelled() {
return future.isCancelled();
}
/**
* Cancel this future and all the previous stage future recursively.
*
* @param mayInterruptIfRunning
* @return
*/
public boolean cancelAll(boolean mayInterruptIfRunning) {
boolean res = true;
if (N.notNullOrEmpty(upFutures)) {
for (CompletableFuture> preFuture : upFutures) {
res = res & preFuture.cancelAll(mayInterruptIfRunning);
}
}
return cancel(mayInterruptIfRunning) && res;
}
/**
* Returns true if this future and all previous stage futures have been recursively cancelled, otherwise false is returned.
*
* @return
*/
public boolean isAllCancelled() {
boolean res = true;
if (N.notNullOrEmpty(upFutures)) {
for (CompletableFuture> preFuture : upFutures) {
res = res & preFuture.isAllCancelled();
}
}
return isCancelled() && res;
}
@Override
public boolean isDone() {
return future.isDone();
}
@Override
public T get() throws InterruptedException, ExecutionException {
return future.get();
}
@Override
public T get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException {
return future.get(timeout, unit);
}
public Pair get2() {
try {
return Pair.of(get(), null);
} catch (Throwable e) {
return Pair.of(null, e);
}
}
public Pair get2(final long timeout, final TimeUnit unit) {
try {
return Pair.of(get(timeout, unit), null);
} catch (Throwable e) {
return Pair.of(null, e);
}
}
public T getNow(T defaultValue) {
try {
return isDone() ? get() : defaultValue;
} catch (InterruptedException | ExecutionException e) {
throw N.toRuntimeException(e);
}
}
public U get(final Function super T, ? extends U> action) {
try {
return action.apply(get());
} catch (InterruptedException | ExecutionException e) {
throw N.toRuntimeException(e);
}
}
public U get(final long timeout, final TimeUnit unit, final Function super T, ? extends U> action) {
try {
return action.apply(get(timeout, unit));
} catch (InterruptedException | ExecutionException | TimeoutException e) {
throw N.toRuntimeException(e);
}
}
public U get(final BiFunction super T, Throwable, ? extends U> action) {
final Pair result = get2();
return action.apply(result.left, result.right);
}
public U get(final long timeout, final TimeUnit unit, final BiFunction super T, Throwable, ? extends U> action) {
final Pair result = get2(timeout, unit);
return action.apply(result.left, result.right);
}
// public void complete() throws InterruptedException, ExecutionException {
// get();
// }
//
// public void complete(Consumer super T> action) {
// try {
// action.accept(get());
// } catch (InterruptedException | ExecutionException e) {
// throw N.toRuntimeException(e);
// }
// }
//
// public void complete(BiConsumer super T, Throwable> action) {
// final Pair result = get2();
// action.accept(result.left, result.right);
// }
CompletableFuture thenApply(final Function super T, ? extends U> action) {
return new CompletableFuture(new Future() {
@Override
public boolean cancel(boolean mayInterruptIfRunning) {
return future.cancel(mayInterruptIfRunning);
}
@Override
public boolean isCancelled() {
return future.isCancelled();
}
@Override
public boolean isDone() {
return future.isDone();
}
@Override
public U get() throws InterruptedException, ExecutionException {
return action.apply(future.get());
}
@Override
public U get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException {
return action.apply(future.get(timeout, unit));
}
}, null, asyncExecutor) {
@Override
public boolean cancelAll(boolean mayInterruptIfRunning) {
return super.cancelAll(mayInterruptIfRunning);
}
@Override
public boolean isAllCancelled() {
return super.isAllCancelled();
}
};
}
// public CompletableFuture thenApply(final BiFunction super T, Throwable, ? extends U> action) {
// return new CompletableFuture(new Future() {
// @Override
// public boolean cancel(boolean mayInterruptIfRunning) {
// return future.cancel(mayInterruptIfRunning);
// }
//
// @Override
// public boolean isCancelled() {
// return future.isCancelled();
// }
//
// @Override
// public boolean isDone() {
// return future.isDone();
// }
//
// @Override
// public U get() throws InterruptedException, ExecutionException {
// final Pair result = get2();
// return action.apply(result.left, result.right);
// }
//
// @Override
// public U get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException {
// final Pair result = get2(timeout, unit);
// return action.apply(result.left, result.right);
// }
// }, null, asyncExecutor) {
// @Override
// public boolean cancelAll(boolean mayInterruptIfRunning) {
// return super.cancelAll(mayInterruptIfRunning);
// }
//
// @Override
// public boolean isAllCancelled() {
// return super.isAllCancelled();
// }
// };
// }
// public CompletableFuture thenAccept(final Consumer super T> action) {
// return thenApply(new Function() {
// @Override
// public Void apply(T t) {
// action.accept(t);
// return null;
// }
// });
// }
//
// public CompletableFuture thenAccept(final BiConsumer super T, Throwable> action) {
// return thenApply(new BiFunction() {
// @Override
// public Void apply(T t, Throwable e) {
// action.accept(t, e);
// return null;
// }
// });
// }
//
// public CompletableFuture thenCombine(final CompletableFuture extends U> other, final BiFunction super T, ? super U, ? extends R> action) {
// return new CompletableFuture(new Future() {
// @Override
// public boolean cancel(boolean mayInterruptIfRunning) {
// return future.cancel(mayInterruptIfRunning) && other.future.cancel(mayInterruptIfRunning);
// }
//
// @Override
// public boolean isCancelled() {
// return future.isCancelled() || other.future.isCancelled();
// }
//
// @Override
// public boolean isDone() {
// return future.isDone() && other.future.isDone();
// }
//
// @Override
// public R get() throws InterruptedException, ExecutionException {
// return action.apply(future.get(), other.future.get());
// }
//
// @Override
// public R get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException {
// final long timeoutInMillis = unit.toMillis(timeout);
// final long now = N.currentMillis();
// final long endTime = timeoutInMillis > Long.MAX_VALUE - now ? Long.MAX_VALUE : now + timeoutInMillis;
//
// final T result = future.get(timeout, unit);
// final U otherResult = other.future.get(N.max(0, endTime - N.currentMillis()), TimeUnit.MILLISECONDS);
//
// return action.apply(result, otherResult);
// }
// }, null, asyncExecutor) {
// @Override
// public boolean cancelAll(boolean mayInterruptIfRunning) {
// return super.cancelAll(mayInterruptIfRunning) && other.cancelAll(mayInterruptIfRunning);
// }
//
// @Override
// public boolean isAllCancelled() {
// return super.isAllCancelled() && other.isAllCancelled();
// }
// };
// }
//
// public CompletableFuture thenCombine(final CompletableFuture extends U> other, final Function, R> action) {
// return new CompletableFuture(new Future() {
// @Override
// public boolean cancel(boolean mayInterruptIfRunning) {
// return future.cancel(mayInterruptIfRunning) && other.future.cancel(mayInterruptIfRunning);
// }
//
// @Override
// public boolean isCancelled() {
// return future.isCancelled() || other.future.isCancelled();
// }
//
// @Override
// public boolean isDone() {
// return future.isDone() && other.future.isDone();
// }
//
// @Override
// public R get() throws InterruptedException, ExecutionException {
// final Pair result = get2();
// final Pair extends U, Throwable> result2 = other.get2();
//
// return action.apply(Tuple.of(result.left, result.right, (U) result2.left, result.right));
// }
//
// @Override
// public R get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException {
// final long timeoutInMillis = unit.toMillis(timeout);
// final long now = N.currentMillis();
// final long endTime = timeoutInMillis > Long.MAX_VALUE - now ? Long.MAX_VALUE : now + timeoutInMillis;
//
// final Pair result = CompletableFuture.this.get2(timeout, unit);
// final Pair extends U, Throwable> result2 = other.get2(N.max(0, endTime - N.currentMillis()), TimeUnit.MILLISECONDS);
//
// return action.apply(Tuple.of(result.left, result.right, (U) result2.left, result.right));
// }
// }, null, asyncExecutor) {
// @Override
// public boolean cancelAll(boolean mayInterruptIfRunning) {
// return super.cancelAll(mayInterruptIfRunning) && other.cancelAll(mayInterruptIfRunning);
// }
//
// @Override
// public boolean isAllCancelled() {
// return super.isAllCancelled() && other.isAllCancelled();
// }
// };
// }
//
// public CompletableFuture thenAcceptBoth(final CompletableFuture extends U> other, final BiConsumer super T, ? super U> action) {
// return thenCombine(other, new BiFunction() {
// @Override
// public Void apply(T t, U u) {
// action.accept(t, u);
// return null;
// }
// });
// }
//
// public CompletableFuture thenAcceptBoth(final CompletableFuture extends U> other, final Consumer> action) {
// return thenCombine(other, new Function, Void>() {
// @Override
// public Void apply(Tuple4 t) {
// action.accept(t);
// return null;
// }
// });
// }
public CompletableFuture thenRun(final Runnable action) {
return execute(new Callable() {
@Override
public Void call() throws Exception {
get();
action.run();
return null;
}
});
}
public CompletableFuture thenRun(final Consumer super T> action) {
return execute(new Callable() {
@Override
public Void call() throws Exception {
action.accept(get());
return null;
}
});
}
public CompletableFuture thenRun(final BiConsumer super T, Throwable> action) {
return execute(new Runnable() {
@Override
public void run() {
final Pair result = get2();
action.accept(result.left, result.right);
}
});
}
public CompletableFuture thenCall(final Try.Callable action) {
return execute(new Callable() {
@Override
public U call() throws Exception {
get();
return action.call();
}
});
}
public CompletableFuture thenCall(final Function super T, R> action) {
return execute(new Callable() {
@Override
public R call() throws Exception {
return action.apply(get());
}
});
}
public CompletableFuture thenCall(final BiFunction super T, Throwable, R> action) {
return execute(new Callable() {
@Override
public R call() throws Exception {
final Pair result = get2();
return action.apply(result.left, result.right);
}
});
}
public CompletableFuture runAfterBoth(final CompletableFuture> other, final Runnable action) {
return execute(new Callable() {
@Override
public Void call() throws Exception {
get();
other.get();
action.run();
return null;
}
}, other);
}
public CompletableFuture runAfterBoth(final CompletableFuture other, final BiConsumer action) {
return execute(new Callable() {
@Override
public Void call() throws Exception {
action.accept(get(), other.get());
return null;
}
}, other);
}
public CompletableFuture runAfterBoth(final CompletableFuture other, final Consumer> action) {
return execute(new Runnable() {
@Override
public void run() {
final Pair result = get2();
final Pair result2 = other.get2();
action.accept(Tuple.of(result.left, result.right, result2.left, result.right));
}
}, other);
}
public CompletableFuture callAfterBoth(final CompletableFuture> other, final Try.Callable action) {
return execute(new Callable() {
@Override
public U call() throws Exception {
get();
other.get();
return action.call();
}
}, other);
}
public CompletableFuture callAfterBoth(final CompletableFuture other, final BiFunction action) {
return execute(new Callable() {
@Override
public R call() throws Exception {
return action.apply(get(), other.get());
}
}, other);
}
public CompletableFuture callAfterBoth(final CompletableFuture other, final Function, R> action) {
return execute(new Callable() {
@Override
public R call() throws Exception {
final Pair result = get2();
final Pair result2 = other.get2();
return action.apply(Tuple.of(result.left, result.right, result2.left, result.right));
}
}, other);
}
public CompletableFuture runAfterEither(final CompletableFuture> other, final Runnable action) {
return execute(new Callable() {
@Override
public Void call() throws Exception {
Futures.anyOf(N.asList(CompletableFuture.this, other)).get();
action.run();
return null;
}
}, other);
}
public CompletableFuture runAfterEither(final CompletableFuture extends T> other, final Consumer super T> action) {
return execute(new Callable() {
@Override
public Void call() throws Exception {
final T result = Futures.anyOf(N.asList(CompletableFuture.this, other)).get();
action.accept(result);
return null;
}
}, other);
}
public CompletableFuture runAfterEither(final CompletableFuture extends T> other, final BiConsumer super T, Throwable> action) {
return execute(new Runnable() {
@Override
public void run() {
final Pair result = Futures.anyOf(N.asList(CompletableFuture.this, other)).get2();
action.accept(result.left, result.right);
}
}, other);
}
public CompletableFuture callAfterEither(final CompletableFuture> other, final Try.Callable action) {
return execute(new Callable() {
@Override
public U call() throws Exception {
Futures.anyOf(N.asList(CompletableFuture.this, other)).get();
return action.call();
}
}, other);
}
public CompletableFuture callAfterEither(final CompletableFuture extends T> other, final Function super T, R> action) {
return execute(new Callable() {
@Override
public R call() throws Exception {
final T result = Futures.anyOf(N.asList(CompletableFuture.this, other)).get();
return action.apply(result);
}
}, other);
}
public CompletableFuture callAfterEither(final CompletableFuture extends T> other, final BiFunction super T, Throwable, R> action) {
return execute(new Callable() {
@Override
public R call() throws Exception {
final Pair result = Futures.anyOf(N.asList(CompletableFuture.this, other)).get2();
return action.apply(result.left, result.right);
}
}, other);
}
// /**
// * Returns a new CompletableFuture that, when either this or the
// * other given CompletableFuture complete normally. If both of the given
// * CompletableFutures complete exceptionally, then the returned
// * CompletableFuture also does so.
// *
// * @param other
// * @param action
// * @return
// */
// public CompletableFuture applyToEither(final CompletableFuture extends T> other, final Function super T, U> action) {
// return Futures.anyOf(N.asList(this, other)).thenApply(action);
// }
//
// /**
// * Returns a new CompletableFuture that, when either this or the
// * other given CompletableFuture complete normally. If both of the given
// * CompletableFutures complete exceptionally, then the returned
// * CompletableFuture also does so.
// *
// * @param other
// * @param action
// * @return
// */
// public CompletableFuture acceptEither(final CompletableFuture extends T> other, final Consumer super T> action) {
// return Futures.anyOf(N.asList(this, other)).thenAccept(action);
// }
// /**
// * Returns a new CompletableFuture that, when this CompletableFuture completes
// * exceptionally, is executed with this CompletableFuture's exception as the
// * argument to the supplied function. Otherwise, if this CompletableFuture
// * completes normally, then the returned CompletableFuture also completes
// * normally with the same value.
// *
// * @param action
// * @return
// */
// public CompletableFuture exceptionally(final Function action) {
// return new CompletableFuture(new Future() {
// @Override
// public boolean cancel(boolean mayInterruptIfRunning) {
// return future.cancel(mayInterruptIfRunning);
// }
//
// @Override
// public boolean isCancelled() {
// return future.isCancelled();
// }
//
// @Override
// public boolean isDone() {
// return future.isDone();
// }
//
// @Override
// public T get() throws InterruptedException, ExecutionException {
// try {
// return future.get();
// } catch (Throwable e) {
// return action.apply(e);
// }
// }
//
// @Override
// public T get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException {
// try {
// return future.get(timeout, unit);
// } catch (Throwable e) {
// return action.apply(e);
// }
// }
// }, null, asyncExecutor) {
// @Override
// public boolean cancelAll(boolean mayInterruptIfRunning) {
// return super.cancelAll(mayInterruptIfRunning);
// }
//
// @Override
// public boolean isAllCancelled() {
// return super.isAllCancelled();
// }
// };
// }
// public CompletableFuture whenComplete(final BiConsumer super T, ? super Throwable> action) {
// return new CompletableFuture<>(new Future() {
// @Override
// public boolean cancel(boolean mayInterruptIfRunning) {
// return future.cancel(mayInterruptIfRunning);
// }
//
// @Override
// public boolean isCancelled() {
// return future.isCancelled();
// }
//
// @Override
// public boolean isDone() {
// return future.isDone();
// }
//
// @Override
// public T get() throws InterruptedException, ExecutionException {
// final Pair result = get2();
//
// if (result.right != null) {
// try {
// action.accept(result.left, result.right);
// } catch (Throwable e) {
// // ignore.
// }
//
// if (result.right instanceof InterruptedException) {
// throw ((InterruptedException) result.right);
// } else if (result.right instanceof ExecutionException) {
// throw ((ExecutionException) result.right);
// } else {
// throw N.toRuntimeException(result.right);
// }
// } else {
// action.accept(result.left, result.right);
// return result.left;
// }
// }
//
// @Override
// public T get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException {
// final Pair result = get2(timeout, unit);
//
// if (result.right != null) {
// try {
// action.accept(result.left, result.right);
// } catch (Throwable e) {
// // ignore.
// }
//
// if (result.right instanceof InterruptedException) {
// throw ((InterruptedException) result.right);
// } else if (result.right instanceof ExecutionException) {
// throw ((ExecutionException) result.right);
// } else {
// throw N.toRuntimeException(result.right);
// }
// } else {
// action.accept(result.left, result.right);
// return result.left;
// }
// }
// }, asyncExecutor);
// }
//
// public CompletableFuture handle(final BiFunction super T, Throwable, ? extends U> action) {
// return new CompletableFuture<>(new Future() {
// @Override
// public boolean cancel(boolean mayInterruptIfRunning) {
// return future.cancel(mayInterruptIfRunning);
// }
//
// @Override
// public boolean isCancelled() {
// return future.isCancelled();
// }
//
// @Override
// public boolean isDone() {
// return future.isDone();
// }
//
// @Override
// public U get() throws InterruptedException, ExecutionException {
// final Pair result = get2();
// return action.apply(result.left, result.right);
// }
//
// @Override
// public U get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException {
// final Pair result = get2(timeout, unit);
// return action.apply(result.left, result.right);
// }
// }, asyncExecutor);
// }
private CompletableFuture execute(final Runnable command) {
return execute(command, null);
}
private CompletableFuture execute(final Runnable command, final CompletableFuture> other) {
return execute(new FutureTask(command, null), other);
}
private CompletableFuture execute(final Callable command) {
return execute(command, null);
}
private CompletableFuture execute(final Callable command, final CompletableFuture> other) {
return execute(new FutureTask<>(command), other);
}
private CompletableFuture execute(final FutureTask futureTask, final CompletableFuture> other) {
asyncExecutor.execute(futureTask);
@SuppressWarnings("rawtypes")
final List> upFutures = other == null ? (List) Arrays.asList(this) : Arrays.asList(this, other);
return new CompletableFuture<>(futureTask, upFutures, asyncExecutor);
}
public CompletableFuture delayed(long delay, TimeUnit unit) {
if (delay <= 0) {
return this;
}
return with(asyncExecutor, delay, TimeUnit.MILLISECONDS);
}
public CompletableFuture with(Executor executor) {
return with(executor, 0, TimeUnit.MILLISECONDS);
}
public CompletableFuture with(final Executor executor, final long delay, final TimeUnit unit) {
N.requireNonNull(executor);
return new CompletableFuture(new Future() {
private final long delayEndTime = N.currentMillis() + unit.toMillis(delay);
private volatile boolean isDelayed = false;
@Override
public boolean cancel(boolean mayInterruptIfRunning) {
return future.cancel(mayInterruptIfRunning);
}
@Override
public boolean isCancelled() {
return future.isCancelled();
}
@Override
public boolean isDone() {
boolean isDone = future.isDone();
if (isDone) {
delay();
}
return isDone;
}
@Override
public T get() throws InterruptedException, ExecutionException {
delay();
return future.get();
}
@Override
public T get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException {
delay();
return future.get(timeout, unit);
}
private void delay() {
if (isDelayed == false) {
isDelayed = true;
N.sleep(delayEndTime - N.currentMillis());
}
}
}, null, executor) {
@Override
public boolean cancelAll(boolean mayInterruptIfRunning) {
return super.cancelAll(mayInterruptIfRunning);
}
@Override
public boolean isAllCancelled() {
return super.isAllCancelled();
}
};
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy