Let the world have no difficult to learn Java
Like, follow, favorites
One, foreword
- Multithreaded programming is common in Java development, from threads and Runnable to Future to CompletableFuture, and the JDK keeps extending the functionality for us with multithreading.
- There are tons of CompletableFuture introductions and tutorials, so why write this article? There are a lot of tutorials, but a lot of copy and paste, or stack of apis. It’s hard to see it in its entirety.
- Therefore, this article mainly talks about CompletableFuture from the perspective of their own learning, and expands CompletableFuture from three main lines, so that you can understand CompletableFuture from a clear level.
Let’s start with the Future mechanic
-
The Future mechanism was introduced in Java 1.5 and represents the result of an asynchronous computation. Refer to Future/Callable
Juejin. Cn/post / 684490… To ensure that you instantly have a sense of layers. I don’t know everything, but I don’t seem to know anything.
-
Future solves the problem of Runnable having no return value by providing two types of get () to get results.
public interface Future<V>{
// block the fetch
V get(a) throws InterruptedException,ExecutionException;
// Wait for timeout to obtain
V get(long timeout,TimeUnit unit) throws InterruptedException,ExecutionException,TimeoutException;
}
Copy the code
Three, CompleteFuture is born
- Blocking the way results are obtained is a clear violation of asynchronous programming, and timeout polling is neither elegant nor timely. Many languages provide callbacks for asynchronous programming, such as Node.js. Java libraries such as Netty and Guava also extend Future to improve the experience of asynchronous programming. Not to be left behind, the authorities have added CompleteFuture and related apis to Java 8, greatly expanding the power of Future.
// Accept Runnable with no return value and use ForkJoinPool.commonPool() thread pool
public static CompletableFuture<Void> runAsync(Runnable runnable)
// Accepts Runnable, with no return value, using the specified Executor thread pool
public static CompletableFuture<Void> runAsync(Runnable runnable, Executor executor)
// Accept Supplier, return U, and use ForkJoinPool.commonPool()
public static <U> CompletableFuture<U> supplyAsync(Supplier<U> supplier)
// Accept Supplier, return U, using the specified executor thread pool
public static <U> CompletableFuture<U> supplyAsync(Supplier<U> supplier, Executor executor)
Copy the code
- You can see that CompleteFuture implements the Future and CompletionStage2 interfaces. A Future, as we all know, is primarily used to implement an asynchronous event that has not been started.
3.1. Create a New CompletableFuture
-
RunAsync has no return value
-
SupplyAsync has a return value
3.2, CompletionStage
-
Stage is the name of the game. As the name implies, the CompletionStage represents a stage in a synchronous or asynchronous calculation, a step in the final result. Multiple completion stages form an assembly line, and when one link is finished, it can be transferred to the next link. A result may need to flow multiple completionstages.
-
For example, in a common exchange scenario, points are deducted first, then SMS is sent, and then personal home page logo is set. In this scenario, points are deducted, SMS is sent, and logo is set respectively in a Stage.
-
ThenApply (). ThenApply (). ThenApply (). One ring at a time.
-
Simple CompletionStage, supplyAsync() After the asynchronous task completes, thenApply () is used to pass the result to the next stage.
Analyze CompletableFuture from four or three perspectives
- With the introduction of CompletableFuture and CompletionStage, we can formally understand CompletableFuture from three perspectives.
4.1 Serial relationship
- thenApply
// This is the simplest and most commonly used method, the chain call of CompletableFuture
public static void main(String[] args) {
CompletableFuture<String> completableFuture = CompletableFuture.runAsync(() -> {
try {
TimeUnit.SECONDS.sleep(3);
System.out.println("hello");
}catch (Exception e){
e.printStackTrace();
}
}).thenApply(s1 -> {
System.out.println(" big");
return s1 + "big";
}).thenApply(s2 -> " world");
}
//hello big world
Copy the code
-
thenRun
- A Runnable is executed when the calculation is complete, instead of using the CompletableFuture calculation, the previous CompletableFuture calculation is ignored and the CompletableFuture type is returned.
-
thenAccept
- ThenApply and thenAccept are similar except that thenAccept is pure consumption and has no return value.
CompletableFuture<Void> completableFuture = CompletableFuture.supplyAsync(() -> {
try {
TimeUnit.SECONDS.sleep(5);
return "success";
} catch (Exception e) {
e.printStackTrace();
return "error";
}
}).thenAcceptAsync(s->{
if ("success".equals(s)){
System.out.println(s);
}else{ System.err.println(s); }});Copy the code
- thenCompose
- Both take a Function, input is the current CompleteableFuture computed value, and return a new CompletableFuture. So this new CompleteableFuture combines the original CompleteableFuture with the CompleteableFuture returned by the function.
- Usually used when the second CompleteableFuture needs to use the result of the first CompleteableFuture as input.
public <U> CompletableFuture<U> thenCompose(Function<? super T, ? extends CompletionStage<U>> fn)
public <U> CompletableFuture<U> thenComposeAsync(Function<? super T, ? extends CompletionStage<U>> fn)
public <U> CompletableFuture<U> thenComposeAsync(Function<? super T, ? extends CompletionStage<U>> fn, Executor executor)
Copy the code
CompletableFuture<Integer> future = CompletableFuture.supplyAsync(() -> 1)
.thenCompose(x -> CompletableFuture.supplyAsync(() -> x+1));
Copy the code
ThenApply and thenCompose both accept a functional interface to a Function, so what's the difference? 1. ThenApply uses the synchronization mapping method. ThenCompose is used in the asynchronous mapping method.
- 🌰 come
//thenApply
thenApply(Function<? super T,? extends U> fn)
//thenCompose
thenCompose(Function<? super T,? extends CompletableFuture<U>> fn)
ThenApply returns an object U, and thenCompose returns CompletableFuture
// From a usage perspective, if your second operation is also asynchronous then use thenCompose. If it is a simple synchronous operation then use thenApply. You should know when to use thenApply and thenCompose a few times.
Copy the code
4.2
-
thenCombine
- Combine, compose According to my level of grade 6, both of them seem to have the meaning of combination and combination. What is the difference?
- The main difference is that the two completableFutures combined by thenCombine have no dependencies, and the second CompletableFuture doesn’t need to wait for the first CompletableFuture to complete execution before starting.
- Although the multiple completableFutures combined by thenCombine are independent, the whole pipeline is synchronized.
🌰 come
// The two completableFutures of thenCombine are not dependencies. The second CompletableFuture is executed before the first CompletableFuture. The final BiFunction() combines two CompletableFuture results.
// Again: the entire pipeline is synchronized
Copy the code
- thenAcceptBoth
// Two completableFutures are processed, and the result is consumed.
// Similar to thenCombine, the 2nd CompletableFuture does not depend on the 1st CompletableFuture to complete and returns the value Void.
Copy the code
- runAfterBoth
- Once both completableFutures are complete, a Runnable is executed. This is similar to thenAcceptBoth and thenCombine, but runAfterBoth doesn’t care about the return value of any CompletableFuture. As long as the CompletableFuture executes it runs, Again, it has no return value.
- allOf
- It simply waits for all compleTableFutures to complete and returns a CompletableFuture.
public static CompletableFuture<Void> allOf(CompletableFuture
... cfs)
Copy the code
public static void main(String[] args) throws Exception {
long start = System.currentTimeMillis();
CompletableFuture<String> completableFuture1 = CompletableFuture.supplyAsync(() -> {
try {
TimeUnit.SECONDS.sleep(1);
System.out.println("i am sleep 1");
} catch (Exception e) {
e.printStackTrace();
}
return "service 1";
});
CompletableFuture<String> completableFuture2 = CompletableFuture.supplyAsync(() -> {
try {
TimeUnit.SECONDS.sleep(2);
System.out.println("i am sleep 2");
} catch (Exception e) {
e.printStackTrace();
}
return "service 2";
});
CompletableFuture<String> completableFuture3 = CompletableFuture.supplyAsync(() -> {
try {
TimeUnit.SECONDS.sleep(3);
System.out.println("i am sleep 3");
} catch (Exception e) {
e.printStackTrace();
}
return "service 3";
});
CompletableFuture<Void> completableFuture = CompletableFuture
.allOf(completableFuture1, completableFuture2, completableFuture3);
completableFuture.join();
System.out.println(System.currentTimeMillis() - start);
}
Copy the code
4.3 OR relationship
- applyToEither
- The result of the fastest completion of the CompletableFuture is input to the next stage
- acceptEither
- The action consumption is executed as soon as the CompletableFuture execution is complete.
- runAfterEither
- Once any CompletableFuture is complete, the next Runnable is executed.
- anyOf
- Any CompletableFuture completes, anyOf returns.
summary
- CompletableFuture greatly expands Future capabilities and simplifies the complexity of asynchronous programming.
- This paper mainly understands the stages between CompletableFuture from the perspectives of AND, OR AND serialization. Different scenes need to be superimposed with different stages.