From 6eb76d2f55964b48f1b3a8078f48cdcc73dbef04 Mon Sep 17 00:00:00 2001 From: huhao80 Date: Thu, 20 Jun 2024 17:00:54 +0800 Subject: [PATCH] =?UTF-8?q?feat:=20=E5=AE=9E=E7=8E=B0`CompletableFutureUti?= =?UTF-8?q?ls`=E4=B8=AD2=E5=88=B05=E4=B8=AA`action`=E7=9A=84`tupleOf*`?= =?UTF-8?q?=E6=96=B9=E6=B3=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../cffu/CompletableFutureUtils.java | 721 +++++++++++++++++- .../cffu/CompletableFutureUtilsTest.java | 76 ++ 2 files changed, 795 insertions(+), 2 deletions(-) diff --git a/cffu-core/src/main/java/io/foldright/cffu/CompletableFutureUtils.java b/cffu-core/src/main/java/io/foldright/cffu/CompletableFutureUtils.java index 1fe4cb25..5a3b0d5c 100644 --- a/cffu-core/src/main/java/io/foldright/cffu/CompletableFutureUtils.java +++ b/cffu-core/src/main/java/io/foldright/cffu/CompletableFutureUtils.java @@ -77,6 +77,183 @@ public static CompletableFuture> mSupplyFastFailAsync( return allResultsOfFastFail(wrapSuppliers(executor, suppliers)); } + /** + * Returns a new CompletableFuture that is asynchronously completed + * by tasks running in the CompletableFuture's default asynchronous execution facility + * with the values obtained by calling the given Suppliers + * in the same order of the given Suppliers arguments. + *

+ * This method is the same as {@link #mSupplyAsync(Supplier[])} except for the fast-fail behavior. + * + * @param supplier1 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier2 the suppliers returning the value to be used to complete the returned CompletableFuture + * @return the new CompletableFuture + * @see #allResultsOfFastFail(CompletionStage[]) + * @see CompletableFuture#supplyAsync(Supplier) + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfMSupplyFastFailAsync(Supplier supplier1,Supplier supplier2) { + return allTupleOfMSupplyFastFailAsync(AsyncPoolHolder.ASYNC_POOL,supplier1,supplier2); + } + + /** + * Returns a new CompletableFuture that is asynchronously completed + * by tasks running in the given Executor with the values obtained by calling the given Suppliers + * in the same order of the given Suppliers arguments. + *

+ * This method is the same as {@link #mSupplyAsync(Executor, Supplier[])} except for the fast-fail behavior. + * + * @param executor the executor to use for asynchronous execution + * @param supplier1 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier2 the suppliers returning the value to be used to complete the returned CompletableFuture + * @return the new CompletableFuture + * @see #allResultsOfFastFail(CompletionStage[]) + * @see CompletableFuture#supplyAsync(Supplier, Executor) + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfMSupplyFastFailAsync( + Executor executor, Supplier supplier1,Supplier supplier2) { + requireNonNull(executor, "executor is null"); + requireArrayAndEleNonNull("supplier", supplier1,supplier2); + return allTupleOf0(wrapSuppliers(executor,supplier1,supplier2), true); + } + + /** + * Returns a new CompletableFuture that is asynchronously completed + * by tasks running in the CompletableFuture's default asynchronous execution facility + * with the values obtained by calling the given Suppliers + * in the same order of the given Suppliers arguments. + *

+ * This method is the same as {@link #mSupplyAsync(Supplier[])} except for the fast-fail behavior. + * + * @param supplier1 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier2 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier3 the suppliers returning the value to be used to complete the returned CompletableFuture + * @return the new CompletableFuture + * @see #allResultsOfFastFail(CompletionStage[]) + * @see CompletableFuture#supplyAsync(Supplier) + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfMSupplyFastFailAsync(Supplier supplier1,Supplier supplier2,Supplier supplier3) { + return allTupleOfMSupplyFastFailAsync(AsyncPoolHolder.ASYNC_POOL,supplier1,supplier2,supplier3); + } + + /** + * Returns a new CompletableFuture that is asynchronously completed + * by tasks running in the given Executor with the values obtained by calling the given Suppliers + * in the same order of the given Suppliers arguments. + *

+ * This method is the same as {@link #mSupplyAsync(Executor, Supplier[])} except for the fast-fail behavior. + * + * @param executor the executor to use for asynchronous execution + * @param supplier1 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier2 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier3 the suppliers returning the value to be used to complete the returned CompletableFuture + * @return the new CompletableFuture + * @see #allResultsOfFastFail(CompletionStage[]) + * @see CompletableFuture#supplyAsync(Supplier, Executor) + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfMSupplyFastFailAsync( + Executor executor, Supplier supplier1,Supplier supplier2,Supplier supplier3) { + requireNonNull(executor, "executor is null"); + requireArrayAndEleNonNull("supplier", supplier1,supplier2,supplier3); + return allTupleOf0(wrapSuppliers(executor,supplier1,supplier2,supplier3), true); + } + + /** + * Returns a new CompletableFuture that is asynchronously completed + * by tasks running in the CompletableFuture's default asynchronous execution facility + * with the values obtained by calling the given Suppliers + * in the same order of the given Suppliers arguments. + *

+ * This method is the same as {@link #mSupplyAsync(Supplier[])} except for the fast-fail behavior. + * + * @param supplier1 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier2 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier3 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier4 the suppliers returning the value to be used to complete the returned CompletableFuture + * @return the new CompletableFuture + * @see #allResultsOfFastFail(CompletionStage[]) + * @see CompletableFuture#supplyAsync(Supplier) + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfMSupplyFastFailAsync(Supplier supplier1,Supplier supplier2,Supplier supplier3,Supplier supplier4) { + return allTupleOfMSupplyFastFailAsync(AsyncPoolHolder.ASYNC_POOL,supplier1,supplier2,supplier3,supplier4); + } + + /** + * Returns a new CompletableFuture that is asynchronously completed + * by tasks running in the given Executor with the values obtained by calling the given Suppliers + * in the same order of the given Suppliers arguments. + *

+ * This method is the same as {@link #mSupplyAsync(Executor, Supplier[])} except for the fast-fail behavior. + * + * @param executor the executor to use for asynchronous execution + * @param supplier1 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier2 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier3 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier4 the suppliers returning the value to be used to complete the returned CompletableFuture + * @return the new CompletableFuture + * @see #allResultsOfFastFail(CompletionStage[]) + * @see CompletableFuture#supplyAsync(Supplier, Executor) + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfMSupplyFastFailAsync( + Executor executor, Supplier supplier1,Supplier supplier2,Supplier supplier3,Supplier supplier4) { + requireNonNull(executor, "executor is null"); + requireArrayAndEleNonNull("supplier", supplier1,supplier2,supplier3,supplier4); + return allTupleOf0(wrapSuppliers(executor,supplier1,supplier2,supplier3,supplier4), true); + } + + + /** + * Returns a new CompletableFuture that is asynchronously completed + * by tasks running in the CompletableFuture's default asynchronous execution facility + * with the values obtained by calling the given Suppliers + * in the same order of the given Suppliers arguments. + *

+ * This method is the same as {@link #mSupplyAsync(Supplier[])} except for the fast-fail behavior. + * + * @param supplier1 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier2 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier3 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier4 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier5 the suppliers returning the value to be used to complete the returned CompletableFuture + * @return the new CompletableFuture + * @see #allResultsOfFastFail(CompletionStage[]) + * @see CompletableFuture#supplyAsync(Supplier) + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfMSupplyFastFailAsync(Supplier supplier1,Supplier supplier2,Supplier supplier3,Supplier supplier4,Supplier supplier5) { + return allTupleOfMSupplyFastFailAsync(AsyncPoolHolder.ASYNC_POOL,supplier1,supplier2,supplier3,supplier4,supplier5); + } + + /** + * Returns a new CompletableFuture that is asynchronously completed + * by tasks running in the given Executor with the values obtained by calling the given Suppliers + * in the same order of the given Suppliers arguments. + *

+ * This method is the same as {@link #mSupplyAsync(Executor, Supplier[])} except for the fast-fail behavior. + * + * @param executor the executor to use for asynchronous execution + * @param supplier1 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier2 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier3 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier4 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier5 the suppliers returning the value to be used to complete the returned CompletableFuture + * @return the new CompletableFuture + * @see #allResultsOfFastFail(CompletionStage[]) + * @see CompletableFuture#supplyAsync(Supplier, Executor) + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfMSupplyFastFailAsync( + Executor executor, Supplier supplier1,Supplier supplier2,Supplier supplier3,Supplier supplier4,Supplier supplier5) { + requireNonNull(executor, "executor is null"); + requireArrayAndEleNonNull("supplier", supplier1,supplier2,supplier3,supplier4,supplier5); + return allTupleOf0(wrapSuppliers(executor,supplier1,supplier2,supplier3,supplier4,supplier5), true); + } + /** * Returns a new CompletableFuture that is asynchronously completed * by tasks running in the CompletableFuture's default asynchronous execution facility @@ -169,6 +346,166 @@ public static CompletableFuture> mSupplyAsync(Executor executor, Sup return allResultsOf(wrapSuppliers(executor, suppliers)); } + /** + * Returns a new CompletableFuture that is asynchronously completed + * by tasks running in the CompletableFuture's default asynchronous execution facility + * with the values obtained by calling the given Suppliers + * in the same order of the given Suppliers arguments. + * + * @param the suppliers' return type + * @return the new CompletableFuture + * @see #allResultsOf(CompletionStage[]) + * @see CompletableFuture#supplyAsync(Supplier) + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfMSupplyAsync(Supplier supplier1,Supplier supplier2) { + return allTupleOfMSupplyAsync(AsyncPoolHolder.ASYNC_POOL,supplier1,supplier2); + } + + + /** + * Returns a new CompletableFuture that is asynchronously completed + * by tasks running in the CompletableFuture's default asynchronous execution facility + * with the values obtained by calling the given Suppliers + * in the same order of the given Suppliers arguments. + * + * @param supplier1 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier2 the suppliers returning the value to be used to complete the returned CompletableFuture + * @return the new CompletableFuture + * @see #allResultsOf(CompletionStage[]) + * @see CompletableFuture#supplyAsync(Supplier) + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfMSupplyAsync(Executor executor, + Supplier supplier1,Supplier supplier2) { + requireNonNull(executor, "executor is null"); + requireArrayAndEleNonNull("supplier", supplier1,supplier2); + return allTupleOf0(wrapSuppliers(executor,supplier1,supplier2), false); + } + + /** + * Returns a new CompletableFuture that is asynchronously completed + * by tasks running in the CompletableFuture's default asynchronous execution facility + * with the values obtained by calling the given Suppliers + * in the same order of the given Suppliers arguments. + * + * @param supplier1 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier2 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier3 the suppliers returning the value to be used to complete the returned CompletableFuture + * @return the new CompletableFuture + * @see #allResultsOf(CompletionStage[]) + * @see CompletableFuture#supplyAsync(Supplier) + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfMSupplyAsync(Supplier supplier1,Supplier supplier2,Supplier supplier3) { + return allTupleOfMSupplyAsync(AsyncPoolHolder.ASYNC_POOL,supplier1,supplier2,supplier3); + } + + /** + * Returns a new CompletableFuture that is asynchronously completed + * by tasks running in the CompletableFuture's default asynchronous execution facility + * with the values obtained by calling the given Suppliers + * in the same order of the given Suppliers arguments. + * + * @param supplier1 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier2 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier3 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param the suppliers' return type + * @return the new CompletableFuture + * @see #allResultsOf(CompletionStage[]) + * @see CompletableFuture#supplyAsync(Supplier) + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfMSupplyAsync(Executor executor,Supplier supplier1,Supplier supplier2,Supplier supplier3) { + requireNonNull(executor, "executor is null"); + requireArrayAndEleNonNull("supplier", supplier1,supplier2,supplier3); + return allTupleOf0(wrapSuppliers(executor,supplier1,supplier2,supplier3), false); + } + + /** + * Returns a new CompletableFuture that is asynchronously completed + * by tasks running in the CompletableFuture's default asynchronous execution facility + * with the values obtained by calling the given Suppliers + * in the same order of the given Suppliers arguments. + * + * @param supplier1 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier2 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier3 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier4 the suppliers returning the value to be used to complete the returned CompletableFuture + * @return the new CompletableFuture + * @see #allResultsOf(CompletionStage[]) + * @see CompletableFuture#supplyAsync(Supplier) + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfMSupplyAsync(Supplier supplier1,Supplier supplier2,Supplier supplier3,Supplier supplier4) { + return allTupleOfMSupplyAsync(AsyncPoolHolder.ASYNC_POOL,supplier1,supplier2,supplier3,supplier4); + } + + /** + * Returns a new CompletableFuture that is asynchronously completed + * by tasks running in the CompletableFuture's default asynchronous execution facility + * with the values obtained by calling the given Suppliers + * in the same order of the given Suppliers arguments. + * + * @param supplier1 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier2 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier3 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier4 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param the suppliers' return type + * @return the new CompletableFuture + * @see #allResultsOf(CompletionStage[]) + * @see CompletableFuture#supplyAsync(Supplier) + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfMSupplyAsync(Executor executor,Supplier supplier1,Supplier supplier2,Supplier supplier3,Supplier supplier4) { + requireNonNull(executor, "executor is null"); + requireArrayAndEleNonNull("supplier", supplier1,supplier2,supplier3,supplier4); + return allTupleOf0(wrapSuppliers(executor,supplier1,supplier2,supplier3,supplier4), false); + } + + /** + * Returns a new CompletableFuture that is asynchronously completed + * by tasks running in the CompletableFuture's default asynchronous execution facility + * with the values obtained by calling the given Suppliers + * in the same order of the given Suppliers arguments. + * + * @param supplier1 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier2 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier3 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier4 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier5 the suppliers returning the value to be used to complete the returned CompletableFuture + * @return the new CompletableFuture + * @see #allResultsOf(CompletionStage[]) + * @see CompletableFuture#supplyAsync(Supplier) + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfMSupplyAsync(Supplier supplier1,Supplier supplier2,Supplier supplier3,Supplier supplier4,Supplier supplier5) { + return allTupleOfMSupplyAsync(AsyncPoolHolder.ASYNC_POOL,supplier1,supplier2,supplier3,supplier4,supplier5); + } + + /** + * Returns a new CompletableFuture that is asynchronously completed + * by tasks running in the CompletableFuture's default asynchronous execution facility + * with the values obtained by calling the given Suppliers + * in the same order of the given Suppliers arguments. + * + * @param supplier1 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier2 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier3 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier4 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param supplier5 the suppliers returning the value to be used to complete the returned CompletableFuture + * @param the suppliers' return type + * @return the new CompletableFuture + * @see #allResultsOf(CompletionStage[]) + * @see CompletableFuture#supplyAsync(Supplier) + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfMSupplyAsync(Executor executor,Supplier supplier1,Supplier supplier2,Supplier supplier3,Supplier supplier4,Supplier supplier5) { + requireNonNull(executor, "executor is null"); + requireArrayAndEleNonNull("supplier", supplier1,supplier2,supplier3,supplier4,supplier5); + return allTupleOf0(wrapSuppliers(executor,supplier1,supplier2,supplier3,supplier4,supplier5), false); + } + @SafeVarargs private static T[] requireArrayAndEleNonNull(String varName, T... array) { requireNonNull(array, varName + "s is null"); @@ -179,7 +516,7 @@ private static T[] requireArrayAndEleNonNull(String varName, T... array) { } private static CompletableFuture[] wrapSuppliers( - Executor executor, Supplier[] suppliers) { + Executor executor, Supplier... suppliers) { @SuppressWarnings("unchecked") CompletableFuture[] cfs = new CompletableFuture[suppliers.length]; for (int i = 0; i < suppliers.length; i++) { @@ -1281,8 +1618,387 @@ public static CompletableFuture> thenMApplyAsync( return toNonMinCf(cf).thenCompose(v -> allResultsOf(wrapFunctions(executor, v, fns))); } + /** + * Returns a new CompletableFuture that, when the given stage completes normally, + * is executed using the CompletableFuture's default asynchronous execution facility, + * with the values obtained by calling the given Functions + * (with the given stage's result as the argument to the given functions) + * in the same order of the given Functions arguments. + * + * @param function1 the functions to use to compute the values of the returned CompletableFuture + * @param function2 the functions to use to compute the values of the returned CompletableFuture + * @param the functions' return type + * @return the new CompletableFuture + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfThenMApplyAsync( + CompletionStage cf,Function function1,Function function2) { + return allTupleOfThenMApplyAsync(cf, AsyncPoolHolder.ASYNC_POOL, function1,function2); + } + + /** + * Returns a new CompletableFuture that, when the given stage completes normally, + * is executed using the given Executor, with the values obtained by calling the given Functions + * (with the given stage's result as the argument to the given functions) + * in the same order of the given Functions arguments. + * + * @param function1 the functions to use to compute the values of the returned CompletableFuture + * @param function2 the functions to use to compute the values of the returned CompletableFuture + * @param the functions' return type + * @return the new CompletableFuture + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfThenMApplyAsync( + CompletionStage cf, Executor executor, Function function1,Function function2) { + requireNonNull(cf, "cf is null"); + requireNonNull(executor, "executor is null"); + requireArrayAndEleNonNull("fn", function1,function2); + CompletableFuture cf1= toNonMinCf(cf); + CompletableFuture completableFuture1 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function1.apply(v), executor)); + CompletableFuture completableFuture2 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function2.apply(v), executor)); + CompletionStage[] css = {completableFuture1,completableFuture2}; + return allTupleOf0(css,false); + } + + /** + * Returns a new CompletableFuture that, when the given stage completes normally, + * is executed using the CompletableFuture's default asynchronous execution facility, + * with the values obtained by calling the given Functions + * (with the given stage's result as the argument to the given functions) + * in the same order of the given Functions arguments. + * + * @param function1 the functions to use to compute the values of the returned CompletableFuture + * @param function2 the functions to use to compute the values of the returned CompletableFuture + * @param function3 the functions to use to compute the values of the returned CompletableFuture + * @return the new CompletableFuture + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfThenMApplyAsync( + CompletionStage cf,Function function1,Function function2,Function function3) { + return allTupleOfThenMApplyAsync(cf, AsyncPoolHolder.ASYNC_POOL, function1,function2,function3); + } + + /** + * Returns a new CompletableFuture that, when the given stage completes normally, + * is executed using the given Executor, with the values obtained by calling the given Functions + * (with the given stage's result as the argument to the given functions) + * in the same order of the given Functions arguments. + * + * @param function1 the functions to use to compute the values of the returned CompletableFuture + * @param function2 the functions to use to compute the values of the returned CompletableFuture + * @param function3 the functions to use to compute the values of the returned CompletableFuture + * @param the functions' return type + * @return the new CompletableFuture + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfThenMApplyAsync( + CompletionStage cf, Executor executor, Function function1,Function function2,Function function3) { + requireNonNull(cf, "cf is null"); + requireNonNull(executor, "executor is null"); + requireArrayAndEleNonNull("fn", function1,function2,function3); + + CompletableFuture cf1= toNonMinCf(cf); + CompletableFuture completableFuture1 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function1.apply(v), executor)); + CompletableFuture completableFuture2 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function2.apply(v), executor)); + CompletableFuture completableFuture3 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function3.apply(v), executor)); + CompletionStage[] css = {completableFuture1,completableFuture2,completableFuture3}; + return allTupleOf0(css,false); + } + + /** + * Returns a new CompletableFuture that, when the given stage completes normally, + * is executed using the CompletableFuture's default asynchronous execution facility, + * with the values obtained by calling the given Functions + * (with the given stage's result as the argument to the given functions) + * in the same order of the given Functions arguments. + * + * @param function1 the functions to use to compute the values of the returned CompletableFuture + * @param function2 the functions to use to compute the values of the returned CompletableFuture + * @param function3 the functions to use to compute the values of the returned CompletableFuture + * @param function4 the functions to use to compute the values of the returned CompletableFuture + * @param the functions' return type + * @return the new CompletableFuture + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfThenMApplyAsync( + CompletionStage cf,Function function1,Function function2,Function function3,Function function4) { + return allTupleOfThenMApplyAsync(cf, AsyncPoolHolder.ASYNC_POOL, function1,function2,function3,function4); + } + + /** + * Returns a new CompletableFuture that, when the given stage completes normally, + * is executed using the given Executor, with the values obtained by calling the given Functions + * (with the given stage's result as the argument to the given functions) + * in the same order of the given Functions arguments. + * + * @param function1 the functions to use to compute the values of the returned CompletableFuture + * @param function2 the functions to use to compute the values of the returned CompletableFuture + * @param function3 the functions to use to compute the values of the returned CompletableFuture + * @param function4 the functions to use to compute the values of the returned CompletableFuture + * @param the functions' return type + * @return the new CompletableFuture + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfThenMApplyAsync( + CompletionStage cf, Executor executor, Function function1,Function function2,Function function3,Function function4) { + requireNonNull(cf, "cf is null"); + requireNonNull(executor, "executor is null"); + requireArrayAndEleNonNull("fn", function1,function2,function3,function4); + + CompletableFuture cf1= toNonMinCf(cf); + CompletableFuture completableFuture1 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function1.apply(v), executor)); + CompletableFuture completableFuture2 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function2.apply(v), executor)); + CompletableFuture completableFuture3 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function3.apply(v), executor)); + CompletableFuture completableFuture4 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function4.apply(v), executor)); + CompletionStage[] css = {completableFuture1,completableFuture2,completableFuture3,completableFuture4}; + return allTupleOf0(css,false); + } + + /** + * Returns a new CompletableFuture that, when the given stage completes normally, + * is executed using the CompletableFuture's default asynchronous execution facility, + * with the values obtained by calling the given Functions + * (with the given stage's result as the argument to the given functions) + * in the same order of the given Functions arguments. + * + * @param function1 the functions to use to compute the values of the returned CompletableFuture + * @param function2 the functions to use to compute the values of the returned CompletableFuture + * @param function3 the functions to use to compute the values of the returned CompletableFuture + * @param function4 the functions to use to compute the values of the returned CompletableFuture + * @param function5 the functions to use to compute the values of the returned CompletableFuture + * @param the functions' return type + * @return the new CompletableFuture + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfThenMApplyAsync( + CompletionStage cf,Function function1,Function function2,Function function3,Function function4,Function function5) { + return allTupleOfThenMApplyAsync(cf, AsyncPoolHolder.ASYNC_POOL, function1,function2,function3,function4,function5); + } + + /** + * Returns a new CompletableFuture that, when the given stage completes normally, + * is executed using the given Executor, with the values obtained by calling the given Functions + * (with the given stage's result as the argument to the given functions) + * in the same order of the given Functions arguments. + * + * @param function1 the functions to use to compute the values of the returned CompletableFuture + * @param function2 the functions to use to compute the values of the returned CompletableFuture + * @param function3 the functions to use to compute the values of the returned CompletableFuture + * @param function4 the functions to use to compute the values of the returned CompletableFuture + * @param function5 the functions to use to compute the values of the returned CompletableFuture + * @param the functions' return type + * @return the new CompletableFuture + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfThenMApplyAsync( + CompletionStage cf, Executor executor, Function function1,Function function2,Function function3,Function function4,Function function5) { + requireNonNull(cf, "cf is null"); + requireNonNull(executor, "executor is null"); + requireArrayAndEleNonNull("fn", function1,function2,function3,function4,function5); + + CompletableFuture cf1= toNonMinCf(cf); + CompletableFuture completableFuture1 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function1.apply(v), executor)); + CompletableFuture completableFuture2 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function2.apply(v), executor)); + CompletableFuture completableFuture3 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function3.apply(v), executor)); + CompletableFuture completableFuture4 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function4.apply(v), executor)); + CompletableFuture completableFuture5 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function5.apply(v), executor)); + CompletionStage[] css = {completableFuture1,completableFuture2,completableFuture3,completableFuture4,completableFuture5}; + return allTupleOf0(css,false); + } + + /** + * Returns a new CompletableFuture that, when the given stage completes normally, + * is executed using the CompletableFuture's default asynchronous execution facility, + * with the values obtained by calling the given Functions + * (with the given stage's result as the argument to the given functions) + * in the same order of the given Functions arguments. + * + * @param function1 the functions to use to compute the values of the returned CompletableFuture + * @param function2 the functions to use to compute the values of the returned CompletableFuture + * @return the new CompletableFuture + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfThenMApplyFastFailAsync( + CompletionStage cf,Function function1,Function function2) { + return allTupleOfThenMApplyFastFailAsync(cf, AsyncPoolHolder.ASYNC_POOL, function1,function2); + } + + /** + * Returns a new CompletableFuture that, when the given stage completes normally, + * is executed using the given Executor, with the values obtained by calling the given Functions + * (with the given stage's result as the argument to the given functions) + * in the same order of the given Functions arguments. + * + * @param function1 the functions to use to compute the values of the returned CompletableFuture + * @param function2 the functions to use to compute the values of the returned CompletableFuture + * @param the functions' return type + * @return the new CompletableFuture + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfThenMApplyFastFailAsync( + CompletionStage cf, Executor executor, Function function1,Function function2) { + requireNonNull(cf, "cf is null"); + requireNonNull(executor, "executor is null"); + requireArrayAndEleNonNull("fn", function1,function2); + CompletableFuture cf1= toNonMinCf(cf); + CompletableFuture completableFuture1 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function1.apply(v), executor)); + CompletableFuture completableFuture2 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function2.apply(v), executor)); + CompletionStage[] css = {completableFuture1,completableFuture2}; + return allTupleOf0(css,true); + } + + /** + * Returns a new CompletableFuture that, when the given stage completes normally, + * is executed using the CompletableFuture's default asynchronous execution facility, + * with the values obtained by calling the given Functions + * (with the given stage's result as the argument to the given functions) + * in the same order of the given Functions arguments. + * + * @param function1 the functions to use to compute the values of the returned CompletableFuture + * @param function2 the functions to use to compute the values of the returned CompletableFuture + * @param function3 the functions to use to compute the values of the returned CompletableFuture + * @param the functions' return type + * @return the new CompletableFuture + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfThenMApplyFastFailAsync( + CompletionStage cf,Function function1,Function function2,Function function3) { + return allTupleOfThenMApplyFastFailAsync(cf, AsyncPoolHolder.ASYNC_POOL, function1,function2,function3); + } + + /** + * Returns a new CompletableFuture that, when the given stage completes normally, + * is executed using the given Executor, with the values obtained by calling the given Functions + * (with the given stage's result as the argument to the given functions) + * in the same order of the given Functions arguments. + * + * @param function1 the functions to use to compute the values of the returned CompletableFuture + * @param function2 the functions to use to compute the values of the returned CompletableFuture + * @param function3 the functions to use to compute the values of the returned CompletableFuture + * @param the functions' return type + * @return the new CompletableFuture + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfThenMApplyFastFailAsync( + CompletionStage cf, Executor executor, Function function1,Function function2,Function function3) { + requireNonNull(cf, "cf is null"); + requireNonNull(executor, "executor is null"); + requireArrayAndEleNonNull("fn", function1,function2,function3); + + CompletableFuture cf1= toNonMinCf(cf); + CompletableFuture completableFuture1 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function1.apply(v), executor)); + CompletableFuture completableFuture2 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function2.apply(v), executor)); + CompletableFuture completableFuture3 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function3.apply(v), executor)); + CompletionStage[] css = {completableFuture1,completableFuture2,completableFuture3}; + return allTupleOf0(css,true); + } + + + /** + * Returns a new CompletableFuture that, when the given stage completes normally, + * is executed using the CompletableFuture's default asynchronous execution facility, + * with the values obtained by calling the given Functions + * (with the given stage's result as the argument to the given functions) + * in the same order of the given Functions arguments. + * + * @param function1 the functions to use to compute the values of the returned CompletableFuture + * @param function2 the functions to use to compute the values of the returned CompletableFuture + * @param function3 the functions to use to compute the values of the returned CompletableFuture + * @param function4 the functions to use to compute the values of the returned CompletableFuture + * @param the functions' return type + * @return the new CompletableFuture + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfThenMApplyFastFailAsync( + CompletionStage cf,Function function1,Function function2,Function function3,Function function4) { + return allTupleOfThenMApplyFastFailAsync(cf, AsyncPoolHolder.ASYNC_POOL, function1,function2,function3,function4); + } + + /** + * Returns a new CompletableFuture that, when the given stage completes normally, + * is executed using the given Executor, with the values obtained by calling the given Functions + * (with the given stage's result as the argument to the given functions) + * in the same order of the given Functions arguments. + * + * @param function1 the functions to use to compute the values of the returned CompletableFuture + * @param function2 the functions to use to compute the values of the returned CompletableFuture + * @param function3 the functions to use to compute the values of the returned CompletableFuture + * @param function4 the functions to use to compute the values of the returned CompletableFuture + * @param the functions' return type + * @return the new CompletableFuture + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfThenMApplyFastFailAsync( + CompletionStage cf, Executor executor, Function function1,Function function2,Function function3,Function function4) { + requireNonNull(cf, "cf is null"); + requireNonNull(executor, "executor is null"); + requireArrayAndEleNonNull("fn", function1,function2,function3,function4); + + CompletableFuture cf1= toNonMinCf(cf); + CompletableFuture completableFuture1 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function1.apply(v), executor)); + CompletableFuture completableFuture2 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function2.apply(v), executor)); + CompletableFuture completableFuture3 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function3.apply(v), executor)); + CompletableFuture completableFuture4 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function4.apply(v), executor)); + CompletionStage[] css = {completableFuture1,completableFuture2,completableFuture3,completableFuture4}; + return allTupleOf0(css,true); + } + + /** + * Returns a new CompletableFuture that, when the given stage completes normally, + * is executed using the CompletableFuture's default asynchronous execution facility, + * with the values obtained by calling the given Functions + * (with the given stage's result as the argument to the given functions) + * in the same order of the given Functions arguments. + * + * @param function1 the functions to use to compute the values of the returned CompletableFuture + * @param function2 the functions to use to compute the values of the returned CompletableFuture + * @param function3 the functions to use to compute the values of the returned CompletableFuture + * @param function4 the functions to use to compute the values of the returned CompletableFuture + * @param function5 the functions to use to compute the values of the returned CompletableFuture + * @param the functions' return type + * @return the new CompletableFuture + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfThenMApplyFastFailAsync( + CompletionStage cf,Function function1,Function function2,Function function3,Function function4,Function function5) { + return allTupleOfThenMApplyFastFailAsync(cf, AsyncPoolHolder.ASYNC_POOL, function1,function2,function3,function4,function5); + } + + /** + * Returns a new CompletableFuture that, when the given stage completes normally, + * is executed using the given Executor, with the values obtained by calling the given Functions + * (with the given stage's result as the argument to the given functions) + * in the same order of the given Functions arguments. + * + * @param function1 the functions to use to compute the values of the returned CompletableFuture + * @param function2 the functions to use to compute the values of the returned CompletableFuture + * @param function3 the functions to use to compute the values of the returned CompletableFuture + * @param function4 the functions to use to compute the values of the returned CompletableFuture + * @param function5 the functions to use to compute the values of the returned CompletableFuture + * @param the functions' return type + * @return the new CompletableFuture + */ + @Contract(pure = true) + public static CompletableFuture> allTupleOfThenMApplyFastFailAsync( + CompletionStage cf, Executor executor, Function function1,Function function2,Function function3,Function function4,Function function5) { + requireNonNull(cf, "cf is null"); + requireNonNull(executor, "executor is null"); + requireArrayAndEleNonNull("fn", function1,function2,function3,function4,function5); + + CompletableFuture cf1= toNonMinCf(cf); + CompletableFuture completableFuture1 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function1.apply(v), executor)); + CompletableFuture completableFuture2 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function2.apply(v), executor)); + CompletableFuture completableFuture3 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function3.apply(v), executor)); + CompletableFuture completableFuture4 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function4.apply(v), executor)); + CompletableFuture completableFuture5 = cf1.thenCompose(v->CompletableFuture.supplyAsync(() -> function5.apply(v), executor)); + CompletionStage[] css = {completableFuture1,completableFuture2,completableFuture3,completableFuture4,completableFuture5}; + return allTupleOf0(css,true); + + } + + private static CompletableFuture[] wrapFunctions( - Executor executor, T v, Function[] fns) { + Executor executor, T v, Function... fns) { @SuppressWarnings("unchecked") CompletableFuture[] cfs = new CompletableFuture[fns.length]; for (int i = 0; i < fns.length; i++) { @@ -1292,6 +2008,7 @@ private static CompletableFuture[] wrapFunctions( return cfs; } + /** * Returns a new CompletableFuture that, when the given stage completes normally, * is executed using the CompletableFuture's default asynchronous execution facility, diff --git a/cffu-core/src/test/java/io/foldright/cffu/CompletableFutureUtilsTest.java b/cffu-core/src/test/java/io/foldright/cffu/CompletableFutureUtilsTest.java index 2da0b91b..92848b85 100644 --- a/cffu-core/src/test/java/io/foldright/cffu/CompletableFutureUtilsTest.java +++ b/cffu-core/src/test/java/io/foldright/cffu/CompletableFutureUtilsTest.java @@ -749,6 +749,82 @@ void test_allTupleOf_NotFastFail() throws Exception { ); } + + @Test + void test_allTupleOfMSupplyAsync() throws Exception { + final Supplier supplier_n = () -> { + sleep(100); + return n; + }; + final Supplier supplier_s = () -> { + sleep(100); + return s; + }; + + final Supplier supplier_d = () -> { + sleep(100); + return d; + }; + final Supplier supplier_an = () -> { + sleep(100); + return anotherN; + }; + final Supplier supplier_nn = () -> { + sleep(100); + return n+n; + }; + assertEquals(Tuple2.of(n, s), allTupleOfMSupplyAsync(supplier_n, supplier_s).get()); + assertEquals(Tuple2.of(n, s), allTupleOfMSupplyFastFailAsync(supplier_n, supplier_s).get()); + + assertEquals(Tuple3.of(n, s, d), allTupleOfMSupplyAsync(supplier_n, supplier_s, supplier_d).get()); + assertEquals(Tuple3.of(n, s, d), allTupleOfMSupplyFastFailAsync(supplier_n, supplier_s, supplier_d).get()); + + assertEquals(Tuple4.of(n, s, d, anotherN), allTupleOfMSupplyAsync(supplier_n, supplier_s, supplier_d, supplier_an).get()); + assertEquals(Tuple4.of(n, s, d, anotherN), allTupleOfMSupplyFastFailAsync(supplier_n, supplier_s, supplier_d, supplier_an).get()); + + assertEquals(Tuple5.of(n, s, d, anotherN, n + n), allTupleOfMSupplyAsync(supplier_n, supplier_s, supplier_d, supplier_an, supplier_nn).get()); + assertEquals(Tuple5.of(n, s, d, anotherN, n + n), allTupleOfMSupplyFastFailAsync(supplier_n, supplier_s, supplier_d, supplier_an, supplier_nn).get()); + } + + + @Test + void test_allTupleOfThenMApplyAsync() throws Exception { + final CompletableFuture completed = completedFuture(n); + final Function function_n = (x) -> { + sleep(100); + return n; + }; + + final Function function_s = (x) -> { + sleep(100); + return s; + }; + + final Function function_d = (x) -> { + sleep(100); + return d; + }; + final Function function_an = (x) -> { + sleep(100); + return anotherN; + }; + final Function function_nn = (x) -> { + sleep(100); + return n+n; + }; + assertEquals(Tuple2.of(n, s), allTupleOfThenMApplyAsync(completed,function_n, function_s).get()); + assertEquals(Tuple2.of(n, s), allTupleOfThenMApplyFastFailAsync(completed,function_n, function_s).get()); + + assertEquals(Tuple3.of(n, s, d), allTupleOfThenMApplyAsync(completed,function_n, function_s, function_d).get()); + assertEquals(Tuple3.of(n, s, d), allTupleOfThenMApplyFastFailAsync(completed,function_n, function_s, function_d).get()); + + assertEquals(Tuple4.of(n, s, d, anotherN), allTupleOfThenMApplyAsync(completed,function_n, function_s, function_d, function_an).get()); + assertEquals(Tuple4.of(n, s, d, anotherN), allTupleOfThenMApplyFastFailAsync(completed,function_n, function_s, function_d, function_an).get()); + + assertEquals(Tuple5.of(n, s, d, anotherN, n + n), allTupleOfThenMApplyAsync(completed,function_n, function_s, function_d, function_an, function_nn).get()); + assertEquals(Tuple5.of(n, s, d, anotherN, n + n), allTupleOfThenMApplyFastFailAsync(completed,function_n, function_s, function_d, function_an, function_nn).get()); + } + @Test void test_mostTupleOfSuccess() throws Exception { final CompletableFuture completed = completedFuture(n);