Issue
Why does this block of code behave differently in Java 8 vs Java 11?
private static String test2() {
CompletableFuture
.runAsync(() -> IntStream.rangeClosed(1, 20).forEach(x -> {
try {
Thread.sleep(500);
System.out.println(x);
} catch (InterruptedException e) {
e.printStackTrace();
}
}));
return "Finish";
}
I expected it to print Finish and then for numbers from 1 to 20 with a 500 ms interval print that number, and then stop execution, and it works correctly in Java 8.
When I run exactly the same method on Java 11, however, it printed Finish and terminated without invoking the runAsync(...) code. I managed to start it by adding an ExecutorService like this
private static String test2() {
final ExecutorService executorService = Executors.newFixedThreadPool(10);
CompletableFuture
.runAsync(() -> IntStream.rangeClosed(1, 10).forEach(x -> {
try {
Thread.sleep(500);
System.out.println(x);
} catch (InterruptedException e) {
e.printStackTrace();
}
}), executorService);
return "Finish";
}
And now it gets executed, but doesn't finish; it gets to 10 and after that just sits without finishing. I figured out how to stop execution by invoking executorService.shutdown();
right before return, but I am 100% sure that this approach is wrong, because usually I will have the same executorService for many methods, and if I shut it down other methods will also fail to execute.
What changed between Java 8 and Java 11 here, and why do I now have to add an explicit executor service, and most importantly how can I finish the method execution properly?
Solution
TL;DR - add ForkJoinPool.commonPool().awaitQuiescence(1000, TimeUnit.SECONDS);
after your call to CompletableFuture.runAsync
and at the end of your code so that System.exit
doesn't stop your runnable. That way you'll get you behavior.
Longer answer:
Okay, so first things first, I tried both examples in Oracles java 8, OpenJDK 8, and OpenJDK 11. Consistent behavior across the board, so my answer is that nothing has changed in these implementations of different java versions that would cause this discrepancy. In both examples, the behavior you see is consistent with what Java tells you it will do.
From the documentation of CompletableFuture.runAsync
Returns a new CompletableFuture that is asynchronously completed by a task running in the
ForkJoinPool.commonPool()
after it runs the given action.
Okay... let's see what ForkJoinPool.commonPool
will tell us (emphasis mine):
Returns the common pool instance. This pool is statically constructed; its run state is unaffected by attempts to
shutdown()
orshutdownNow()
. However this pool and any ongoing processing are automatically terminated upon programSystem.exit(int)
. Any program that relies on asynchronous task processing to complete before program termination should invokecommonPool().awaitQuiescence
, before exit.
Aha, so that's why we don't see the countdown when using the common pool, it's because the common pool will be terminated upon system exit, which is exactly what happens when we return from the method and exit the program (assuming your example is really that simple as you show it.... like with a single method call in main
... anyways)
So why would the custom executor work? Because, as you've already noticed, that executor has not been terminated. There is still a piece of code running in the background, although idly, that Java doesn't have the power to stop.
So what can we do now?
One option is to do our own executor and shut it down once we are done, much like you have suggested. I would argue that this approach isn't all that bad after all to use.
Second option is to follow what the java doc says.
Any program that relies on asynchronous task processing to complete before program termination should invoke
commonPool().awaitQuiescence
, before exit.
public boolean awaitQuiescence(long timeout, TimeUnit unit)
If called by a ForkJoinTask operating in this pool, equivalent in effect to ForkJoinTask.helpQuiesce(). Otherwise, waits and/or attempts to assist performing tasks until this pool isQuiescent() or the indicated timeout elapses.
So we can call that method and specify a timeout for all common processes in the common pool. My opinion is that this is somewhat business specific, since now you have to answer the question - What the heck should the timeout be now??.
Third option is to use the power of CompletableFuture
s and hoist this runAsync
method to a variable:
CompletableFuture<Void> voidCompletableFuture = CompletableFuture.runAsync(() -> ...
...
...
bla bla bla code bla bla bla
...
...
voidCompletableFuture.join();
// or if you want to handle exceptions, use get
voidCompletableFuture.get();
and then right when you need it, you join()/get()
whatever you need to have as a return value. I prefer this the most since the code is most clean and understandable like this. Also I can chain my CFs all I want and do funky stuff with them.
In the case where you don't need a return value and don't need to do anything else and just want a simple return of a string and async processing of counting from 1 to 20, then just shove ForkJoinPool.commonPool().awaitQuiescence(1000, TimeUnit.SECONDS);
somewhere convenient to you and give it some ridiculous timeout, thus guaranteeing you'll exit upon all idle processes.
Answered By - mnestorov
Answer Checked By - Candace Johnson (JavaFixing Volunteer)