Using Java 8 CompletableFuture and Rx-Java Observable
A simple scatter-gather scenario using Java 8 CompletableFuture and using Rx-Java Observable.
Join the DZone community and get the full member experience.
Join For FreeThe scenario is simple - Spawn about 10 tasks, each returning a string, and ultimately collect the results into a list.
Sequential
A sequential version of this would be the following:
public void testSequentialScatterGather() throws Exception {
List<String> list =
IntStream.range(0, 10)
.boxed()
.map(this::generateTask)
.collect(Collectors.toList());
logger.info(list.toString());
}
private String generateTask(int i) {
Util.delay(2000);
return i + "-" + "test";
}
With CompletableFuture
A method can be made to return a CompletableFuture using a utility method called supplyAsync, I am using a variation of this method which accepts an explicit Executor to use, also I am deliberately throwing an exception for one of the inputs:
private CompletableFuture<String> generateTask(int i,
ExecutorService executorService) {
return CompletableFuture.supplyAsync(() -> {
Util.delay(2000);
if (i == 5) {
throw new RuntimeException("Run, it is a 5!");
}
return i + "-" + "test";
}, executorService);
}
Now to scatter the tasks:
List<CompletableFuture<String>> futures =
IntStream.range(0, 10)
.boxed()
.map(i -> this.generateTask(i, executors).exceptionally(t -> t.getMessage()))
.collect(Collectors.toList());
At the end of scattering the tasks the result is a list of CompletableFuture. Now, to obtain the list of String from this is a little tricky, here I am using one of the solutions suggested in Stackoverflow:
CompletableFuture<List<String>> result = CompletableFuture.allOf(futures.toArray(new CompletableFuture[futures.size()]))
.thenApply(v -> futures.stream()
.map(CompletableFuture::join)
.collect(Collectors.toList()));
CompletableFuture.allOf method is being used here purely to compose the next action to take once all the scattered tasks are completed, once the tasks are completed the futures are again streamed and collected into a list of string.
The final result can then be presented asynchronously:
result.thenAccept(l -> {
logger.info(l.toString());
});
With Rx-java Observable
Scatter gather with Rx-java is relatively cleaner than the CompletableFuture version as Rx-java provides better ways to compose the results together, again the method which performs the scattered task:
private Observable<String> generateTask(int i, ExecutorService executorService) {
return Observable
.<String>create(s -> {
Util.delay(2000);
if ( i == 5) {
throw new RuntimeException("Run, it is a 5!");
}
s.onNext( i + "-test");
s.onCompleted();
}).onErrorReturn(e -> e.getMessage()).subscribeOn(Schedulers.from(executorService));
}
and to scatter the tasks:
List<Observable<String>> obs =
IntStream.range(0, 10)
.boxed()
.map(i -> generateTask(i, executors)).collect(Collectors.toList());
Once more I have a List of Observable's, and what I need is a List of results, Observable provides a merge method to do just that:
Observable<List<String>> merged = Observable.merge(obs).toList();
which can be subscribed to and the results printed when available:
merged.subscribe(
l -> logger.info(l.toString()));
Published at DZone with permission of Biju Kunjummen, DZone MVB. See the original article here.
Opinions expressed by DZone contributors are their own.
Comments