简单介绍一下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(...) 前面出错时触发,返回一个安全的兜底值
相关推荐
长孙豪翔5 小时前
在.net中读写config文件的各种方法
java·数据库·.net
tachibana25 小时前
hot100 回文链表(234)
java·网络·数据结构·leetcode·链表
可乐ea5 小时前
【Java八股|第10篇】Java 中的包装类和自动拆装箱
java·面试题·包装类·java八股
zfoo-framework5 小时前
mongo最佳实战(from mongo中文社区)
java
深盾科技_Virbox6 小时前
加密狗授权能力选型:从授权模型到全生命周期管理
java·网络·数据库
. . . . .7 小时前
Egg框架深入
java·开发语言
RainCity7 小时前
Java Swing 自定义组件库分享(十三)
java·笔记·后端
livemetee8 小时前
【关于Spring声明式事务】
java·后端·spring
倒流时光三十年8 小时前
Java 内存模型(JMM)通俗解释
java·开发语言
码兄科技8 小时前
Java AI智能体开发实战:从零构建企业级智能应用指南
java·开发语言·人工智能