0

0

Java中如何用CompletableFuture处理异步任务

穿越時空

穿越時空

发布时间:2025-06-23 17:57:02

|

725人浏览过

|

来源于php中文网

原创

java中completablefuture用于处理异步任务,提高程序响应性和效率。1.它代表异步计算结果,支持链式调用定义后续操作;2.通过supplyasync/runasync创建异步任务,completedfuture创建已完成任务;3.使用thenapply/thenaccept/thenrun实现链式操作,thencombine/thencompose组合多个任务;4.提供allof/anyof编排多任务执行;5.通过exceptionally/handle/whencomplete处理异常;6.相比future,其优势是非阻塞回调、丰富组合能力、简化线程管理;7.选择executorservice需根据任务类型决定,如forkjoinpool适用于cpu密集型任务;8.get()方法会阻塞,应优先使用then系列方法避免阻塞;9.使用ortimeout设置超时,配合exceptionally处理超时异常;10.cancel()方法可取消任务,但不一定能立即停止执行;11.与spring集成可通过@async声明异步方法,返回completablefuture。

Java中如何用CompletableFuture处理异步任务

Java中,CompletableFuture提供了一种强大的方式来处理异步任务,它允许你组合、编排和管理异步操作,从而提高程序的响应性和效率。

Java中如何用CompletableFuture处理异步任务

解决方案

Java中如何用CompletableFuture处理异步任务

CompletableFuture的核心在于它代表了一个异步计算的结果,你可以链式调用它的方法来定义任务完成后的后续操作。以下是一些关键用法:

立即学习Java免费学习笔记(深入)”;

  1. 创建CompletableFuture:

    Java中如何用CompletableFuture处理异步任务
    • CompletableFuture.supplyAsync(() -> ...): 用于执行一个有返回值的异步任务。
    • CompletableFuture.runAsync(() -> ...): 用于执行一个没有返回值的异步任务。
    • CompletableFuture.completedFuture(value): 创建一个已经完成的CompletableFuture,直接返回指定的值。
    // 异步执行一个耗时操作,返回字符串
    CompletableFuture future = CompletableFuture.supplyAsync(() -> {
        // 模拟耗时操作
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
            throw new RuntimeException(e);
        }
        return "Hello, Async World!";
    });
    
    // 异步执行一个无返回值操作
    CompletableFuture voidFuture = CompletableFuture.runAsync(() -> {
        System.out.println("Running async task...");
    });
  2. 链式操作:

    • thenApply(Function): 当CompletableFuture完成时,将结果传递给Function进行处理,返回一个新的CompletableFuture,包含Function的返回值。
    • thenAccept(Consumer): 当CompletableFuture完成时,将结果传递给Consumer进行处理,没有返回值。
    • thenRun(Runnable): 当CompletableFuture完成时,执行Runnable,没有返回值,也不关心CompletableFuture的结果。
    • thenCombine(CompletableFuture, BiFunction): 当两个CompletableFuture都完成时,将它们的结果传递给BiFunction进行处理,返回一个新的CompletableFuture,包含BiFunction的返回值。
    • thenCompose(Function): 类似于thenApply,但Function的返回值必须是另一个CompletableFuture,用于扁平化嵌套的CompletableFuture。
    • exceptionally(Function): 当CompletableFuture抛出异常时,使用Function处理异常,返回一个新的CompletableFuture,包含Function的返回值。
    future.thenApply(result -> result.toUpperCase()) // 将结果转换为大写
          .thenAccept(upperCaseResult -> System.out.println("Result: " + upperCaseResult)) // 打印结果
          .exceptionally(ex -> {
              System.err.println("Error: " + ex.getMessage());
              return null; // 或者返回一个默认值
          });
  3. 组合多个CompletableFuture:

    • allOf(CompletableFuture...): 等待所有CompletableFuture完成。
    • anyOf(CompletableFuture...): 只要有一个CompletableFuture完成,就返回。
    CompletableFuture future1 = CompletableFuture.supplyAsync(() -> "Task 1");
    CompletableFuture future2 = CompletableFuture.supplyAsync(() -> "Task 2");
    
    CompletableFuture allFutures = CompletableFuture.allOf(future1, future2);
    
    allFutures.thenRun(() -> {
        System.out.println("All tasks completed!");
        try {
            System.out.println("Result 1: " + future1.get());
            System.out.println("Result 2: " + future2.get());
        } catch (InterruptedException | ExecutionException e) {
            e.printStackTrace();
        }
    });
  4. 异常处理:

    • exceptionally(Function): 处理异常,返回一个备用值。
    • handle(BiFunction): 处理结果或异常,返回一个新值。
    • whenComplete(BiConsumer): 在完成时执行,无论成功还是失败,不修改结果。
    CompletableFuture futureWithException = CompletableFuture.supplyAsync(() -> {
        throw new RuntimeException("Something went wrong!");
    });
    
    futureWithException.exceptionally(ex -> {
        System.err.println("Exception handled: " + ex.getMessage());
        return "Default Value";
    }).thenAccept(result -> System.out.println("Result: " + result));

CompletableFuture相比Future的优势是什么?

CompletableFuture最大的优势在于它提供了非阻塞的API,允许你注册回调函数,在任务完成时自动执行,而不需要像Future那样通过get()方法阻塞等待结果。 此外,CompletableFuture提供了更丰富的组合和编排能力,可以方便地构建复杂的异步流程。 Future需要手动进行线程管理,而CompletableFuture可以使用ForkJoinPool或者自定义的ExecutorService,简化了线程的管理。

Pascal基础教程 Pascal入门必备基础教程 CHM版
Pascal基础教程 Pascal入门必备基础教程 CHM版

无论做任何事情,都要有一定的方式方法与处理步骤。计算机程序设计比日常生活中的事务处理更具有严谨性、规范性、可行性。为了使计算机有效地解决某些问题,须将处理步骤编排好,用计算机语言组成“序列”,让计算机自动识别并执行这个用计算机语言组成的“序列”,完成预定的任务。将处理问题的步骤编排好,用计算机语言组成序列,也就是常说的编写程序。在Pascal语言中,执行每条语句都是由计算机完成相应的操作。编写Pascal程序,是利用Pasca

下载

如何选择合适的ExecutorService?

选择合适的ExecutorService取决于你的应用场景。 如果是CPU密集型任务,可以使用ForkJoinPool.commonPool()或者创建一个固定大小的线程池。 如果是IO密集型任务,可以创建一个CachedThreadPool或者一个固定大小的线程池,并根据实际情况调整线程池的大小。 重要的是要监控线程池的使用情况,避免线程饥饿或线程过多导致性能下降。 也可以使用Executors.newVirtualThreadPerTaskExecutor()来创建虚拟线程池,但需要JDK21及以上。

CompletableFuture的get()方法会阻塞吗?如何避免阻塞?

CompletableFuture.get()方法会阻塞当前线程,直到结果可用或超时。 为了避免阻塞,应该尽量使用非阻塞的回调方法,例如thenApplythenAcceptthenRun等。 如果必须使用get()方法,可以设置超时时间,例如get(timeout, TimeUnit),并在超时后进行处理。 另一种方法是使用join()方法,它会抛出一个未经检查的异常,不需要显式处理InterruptedExceptionExecutionException

如何处理CompletableFuture中的超时?

可以使用orTimeout(long, TimeUnit)方法来设置超时时间。 如果CompletableFuture在指定时间内没有完成,会抛出一个TimeoutException。 你可以使用exceptionally方法来处理这个异常。

CompletableFuture futureWithTimeout = CompletableFuture.supplyAsync(() -> {
    try {
        Thread.sleep(2000); // 模拟耗时操作
    } catch (InterruptedException e) {
        Thread.currentThread().interrupt();
        throw new RuntimeException(e);
    }
    return "Result";
}).orTimeout(1, TimeUnit.SECONDS);

futureWithTimeout.thenAccept(result -> System.out.println("Result: " + result))
                 .exceptionally(ex -> {
                     System.err.println("Timeout Exception: " + ex.getMessage());
                     return null;
                 });

CompletableFuture的cancel()方法有什么作用?

CompletableFuture.cancel(boolean mayInterruptIfRunning)方法用于取消CompletableFuture的执行。 mayInterruptIfRunning参数指定是否中断正在运行的任务。 如果任务已经完成或已经被取消,调用此方法不会有任何效果。 取消一个CompletableFuture会使其抛出一个CancellationException。 需要注意的是,取消操作不一定能立即停止任务的执行,这取决于任务的具体实现。

CompletableFuture如何与Spring框架集成?

Spring框架提供了对CompletableFuture的良好支持。 你可以使用@Async注解将一个方法声明为异步方法,Spring会自动将其放入一个线程池中执行。 你还可以使用AsyncResult类来封装异步方法的返回值,并将其转换为CompletableFuture。

@Service
public class AsyncService {

    @Async
    public CompletableFuture doSomethingAsync() {
        // 执行异步操作
        return CompletableFuture.completedFuture("Async Result");
    }
}

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

832

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

737

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

734

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

398

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16925

2023.08.03

Java 桌面应用开发(JavaFX 实战)
Java 桌面应用开发(JavaFX 实战)

本专题系统讲解 Java 在桌面应用开发领域的实战应用,重点围绕 JavaFX 框架,涵盖界面布局、控件使用、事件处理、FXML、样式美化(CSS)、多线程与UI响应优化,以及桌面应用的打包与发布。通过完整示例项目,帮助学习者掌握 使用 Java 构建现代化、跨平台桌面应用程序的核心能力。

36

2026.01.14

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.5万人学习

C# 教程
C# 教程

共94课时 | 6.7万人学习

Java 教程
Java 教程

共578课时 | 46万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号