文章目录
- 1 什么是 CompletableFuture?
- 2 如何正确使用 CompletableFuture 对象?
- 3 如何结合回调函数处理异步任务结果?
- 4 如何组合并处理多个 CompletableFuture?
1 什么是 CompletableFuture?
CompletableFuture 是 Java 8 引入的一个强大的异步编程工具。允许以声明式的方式处理异步任务的结果,避免了传统回调和手动管理线程的复杂性。
CompletableFuture 可以组合和链式调用,高效地利用多核处理器的能力,并且减少了传统并发编程中常见的竞态条件和死锁等问题。
在日常开发中,经常需要处理那些可能耗时的任务,比如网络请求、数据库查询或者复杂的计算。使用 CompletableFuture,可以告诉程序如何在后台执行这些任务,然后在任务完成后执行特定的操作。
可以想象一下,CompletableFuture 就像是一条可以穿越时间的信使,你可以把一项任务托付给它,然后继续做其他事情。当任务完成时,它会及时将结果送回来,让你可以立即处理。这样,你就不必在等待任务完成的过程中浪费时间,而是可以更高效地利用自己的资源。
2 如何正确使用 CompletableFuture 对象?
CompletableFuture 可以以一种非阻塞的方式执行异步任务,并能够在任务完成后立即得到通知。通过链式调用的方式,可以很方便地组合多个异步操作,处理它们的结果或者异常。
通过 CompletableFuture.supplyAsync()
方法创建一个 CompletableFuture 对象,并指定一个需要异步执行的任务:
// Supplier 函数会在一个新的线程上异步执行
CompletableFuture<String> future = CompletableFuture.supplyAsync(() -> {
// 模拟一个耗时操作,如从数据库中读取数据
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
}
return "异步任务完成";
});
通过在 CompletableFuture 上添加一些操作,比如处理任务的结果或者处理任务执行过程中可能发生的异常:
// thenAccept()方法接收一个 Consumer 函数
future.thenAccept(result -> {
System.out.println("任务完成,结果为:" + result);
}).exceptionally(ex -> {
System.out.println("任务出现异常:" + ex.getMessage());
return null;
});
3 如何结合回调函数处理异步任务结果?
结合回调函数处理异步任务结果的过程可以比作在等待一份重要的快递时安排一个通知服务。这个通知服务就是回调函数,它会在快递送达时通知你,或者在处理完成后执行特定的操作。
在 Java 的 CompletableFuture
中,这种模式可以通过 supplyAsync()
、thenApply()
、thenAccept()
和 handle()
方法来实现。
创建一个异步任务时,使用 CompletableFuture.supplyAsync()
可以启动一个任务,这个任务在后台线程中执行,直到它完成。假设有一个任务需要从远程服务器获取数据:
// supplyAsync() 方法接收一个 Supplier 函数,这个函数会在后台线程中运行,并返回一个结果
// 结果会被封装在 CompletableFuture 对象中,等待进一步处理
CompletableFuture<String> future = CompletableFuture.supplyAsync(() -> {
// 模拟从远程服务器获取数据
try {
Thread.sleep(3000);
} catch (InterruptedException e) {
e.printStackTrace();
}
return "数据已成功获取";
});
使用 thenApply()
方法,可以在异步任务完成后,对结果进行转换。这个方法接收一个 Function
函数,这个函数会接收任务的结果,并返回一个新结果。比如,将获取的数据进行处理:
// thenApply() 方法将原始数据转换为大写形式
// 处理后的结果会成为新的 CompletableFuture 对象的结果
CompletableFuture<String> processedFuture = future.thenApply(result -> {
// 对结果进行处理
return result.toUpperCase();
});
为了执行一个操作而不关心处理的结果,可以使用 thenAccept()
方法。这个方法接收一个 Consumer
函数,它处理任务完成时的结果,可以在异步任务完成时执行一些操作,比如日志记录或通知用户。例如,将结果打印到控制台:
// 在任务完成后会调用传入的 Consumer 函数,并将结果传递给它
future.thenAccept(result -> {
System.out.println("任务完成,结果是:" + result);
});
在任务执行过程中,可能会遇到异常。handle()
方法可以用来处理这些异常,它接收一个 BiFunction
函数,这个函数接收结果和异常(如果有的话),并返回一个处理后的结果。例如:
// handle() 方法检查是否有异常发生
// 如果有异常,它会处理异常并返回一个默认的结果
// 如果没有异常,它会处理正常的结果
CompletableFuture<String> handledFuture = future.handle((result, ex) -> {
if (ex != null) {
// 处理异常
System.out.println("任务发生错误:" + ex.getMessage());
return "错误处理结果";
}
// 处理正常结果
return result.toLowerCase();
});
对于这四种回调函数,可以使得异步任务的结果处理变得灵活而强大。通过结合使用不同的回调函数,可以对异步任务的结果进行多种操作,保证程序在处理复杂任务时仍然保持清晰和高效。
4 如何组合并处理多个 CompletableFuture?
组合和处理多个 CompletableFuture
可以让并发任务变得更加灵活和高效。设想有多个任务需要并行执行,然后将它们的结果结合起来进行进一步处理。
在进行组合时,最基本的方法之一是将多个 CompletableFuture
的结果合并。比如,有两个任务需要并行完成,获取两个不同的数据源,然后将这两个结果结合起来。
可以使用 thenCombine()
方法,它接收两个 CompletableFuture
和一个合并函数,两个 CompletableFuture
必须在相同的线程池中执行。
假设有两个任务分别从不同的 API 获取数据:
CompletableFuture<String> future1 = CompletableFuture.supplyAsync(() -> {
// 模拟从第一个 API 获取数据
return "数据1";
});
CompletableFuture<String> future2 = CompletableFuture.supplyAsync(() -> {
// 模拟从第二个 API 获取数据
return "数据2";
});
为了将这两个结果结合起来,可以使用 thenCombine()
:
// thenCombine() 方法接收两个 CompletableFuture 和一个函数,这个函数将两个任务的结果合并成一个结果
// 最终的结果是将两个字符串连接在一起
CompletableFuture<String> combinedFuture = future1.thenCombine(future2, (result1, result2) -> {
// 将两个结果结合成一个
return result1 + " 和 " + result2;
});
另一个有用的方法是 allOf()
方法。当有多个任务需要并行执行,并且在所有任务完成后执行某个操作时,allOf()
非常有用。它接收一个 CompletableFuture
数组,并在所有这些 CompletableFuture
完成时触发。可以用来等待多个异步任务完成,然后执行某个操作:
CompletableFuture<Void> allOfFuture = CompletableFuture.allOf(future1, future2);
要获取所有任务的结果,可以在 allOf()
的结果上添加一个回调函数:
// thenRun() 方法会在所有任务完成后执行,它不需要处理结果,只是执行某个操作
allOfFuture.thenRun(() -> {
// 处理所有任务完成后的操作
try {
String result1 = future1.get();
String result2 = future2.get();
System.out.println("任务1的结果: " + result1);
System.out.println("任务2的结果: " + result2);
} catch (InterruptedException | ExecutionException e) {
e.printStackTrace();
}
});
如果有任务依赖于另一个任务的结果,可以使用 thenCompose()
方法。这种情况下,第二个任务会在第一个任务完成后开始执行。thenCompose()
方法接收一个返回 CompletableFuture
的函数,然后将这两个 CompletableFuture
链接起来:
CompletableFuture<String> future3 = future1.thenCompose(result1 -> {
// 使用第一个任务的结果来创建新的 CompletableFuture
return CompletableFuture.supplyAsync(() -> result1 + " 处理完成");
});
对于这些方法来说,它们为处理多个异步任务提供了强大的工具,使得并发编程更加高效和灵活。通过合理使用这些方法,可以实现复杂的异步任务组合和处理逻辑,确保程序的高效执行。
世界会向那些有目标和远见的人让路