简单介绍一下CompletableFuture,从最简单的用法介绍,

文章目录

      • [1. 最简单的创建方式:提交一个异步任务](#1. 最简单的创建方式:提交一个异步任务)
      • [2. 升级玩法:做完一件事,自动做下一件事(链式调用)](#2. 升级玩法:做完一件事,自动做下一件事(链式调用))
        • [`thenApply` ------ 转换结果](#thenApply —— 转换结果)
        • [`thenAccept` ------ 消费结果](#thenAccept —— 消费结果)
      • [3. 应对意外:异常处理](#3. 应对意外:异常处理)
      • 核心总结

在 Java 的并发编程中,CompletableFuture 就像是一个承诺(Promise)。它代表了一个异步计算的结果:这个结果现在可能还没好,但等会儿好了,它会自动通知你,或者直接帮你执行下一步。

在它出现之前(Java 8 之前),我们用 Future,但它有个致命的大坑------你必须死等它结束(调用 .get() 会阻塞线程),或者不停地去轮询。而 CompletableFuture 彻底解放了我们,让我们可以用链式调用写出优雅的异步代码。


1. 最简单的创建方式:提交一个异步任务

如果你只是想把一个耗时的任务扔到后台去执行,不想卡住主线程,最常用的是 runAsync(无返回值)和 supplyAsync(有返回值)。

java 复制代码
import java.util.concurrent.CompletableFuture;

public class SimpleDemo {
    public static void main(String[] args) throws Exception {
        // 1. 扔一个有返回值的异步任务给后台(默认在 ForkJoinPool 线程池执行)
        CompletableFuture<String> future = CompletableFuture.supplyAsync(() -> {
            try { Thread.sleep(2000); } catch (InterruptedException e) {} // 模拟耗时 2 秒
            return "做好了!";
        });

        System.out.println("主线程继续做别的事...");

        // 2. 强行获取结果(这步会阻塞,直到后台做完)
        String result = future.get(); 
        System.out.println("拿到结果: " + result);
    }
}

2. 升级玩法:做完一件事,自动做下一件事(链式调用)

CompletableFuture 最厉害的地方在于,你可以像接水管一样,把好几个步骤串联起来。前一步的结果会自动传给下一步

thenApply ------ 转换结果

收到上一步的结果,处理一下,再返回一个新结果。

java 复制代码
CompletableFuture.supplyAsync(() -> "hello")
    .thenApply(result -> result + " world")   // 收到 "hello",加工成 "hello world"
    .thenApply(String::toUpperCase)          // 收到 "hello world",变成大写
    .thenAccept(System.out::println);        // 最终打印: HELLO WORLD
thenAccept ------ 消费结果

收到上一步的结果,把它用掉,没有返回值(通常作为链路的终点)。

java 复制代码
CompletableFuture.supplyAsync(() -> "外卖到了")
    .thenAccept(food -> System.out.println("开吃: " + food));

3. 应对意外:异常处理

异步代码最怕执行到一半悄悄报错,你还不知道。CompletableFuture 提供了很温柔的异常捕获机制:exceptionally。它就像 try-catch,如果前面任何一步崩了,都会掉进这里,并且可以给一个兜底的默认值

java 复制代码
CompletableFuture.supplyAsync(() -> {
        if (true) { throw new RuntimeException("厨房着火了!"); }
        return "做好的菜";
    })
    .exceptionally(ex -> {
        System.out.println("发生异常: " + ex.getMessage());
        return "方便面(兜底方案)"; // 报错了就吃方便面
    })
    .thenAccept(food -> System.out.println("最终食物: " + food));

核心总结

你可以把它的常用 API 分成三类来记:

动作类型 API 命名规律 特点
开启任务 supplyAsync(...) / runAsync(...) 把任务扔进后台线程池
接续处理 thenApply(...) 拿到上一步的结果,加工并返回新结果
收尾消费 thenAccept(...) 拿到上一步的结果,消费掉,不返回新结果
救场兜底 exceptionally(...) 前面出错时触发,返回一个安全的兜底值
相关推荐
zandy101110 小时前
2026 BI平台安全治理体系构建:从权限模型到零信任架构
java·开发语言
SuniaWang10 小时前
《Agentx专栏》02-技术选型:预算有限时如何做出正确的技术决策
java·spring·架构·langchain·milvus·agenx·opl
羡寒.10 小时前
接口突然变慢,你怎么排查?
java·后端·spring
zuowei288910 小时前
编程语言对比:C/C++/Java/C#/PHP
java·c语言·c++
百数平台10 小时前
功能更新——百数详情页“数据简报”与“关联标签页”配置指南
java·服务器·前端
接着奏乐接着舞10 小时前
java lambda表达式
java·开发语言·python
金銀銅鐵10 小时前
[Java] 自己写程序,来解析字段的 descriptor
java·后端
ch.ju10 小时前
Java程序设计(第3版)第四章——成员方法
java·开发语言
Dicky-_-zhang11 小时前
微服务安全防护实战:OAuth2与JWT鉴权
java·jvm