简单介绍一下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(...) 前面出错时触发,返回一个安全的兜底值
相关推荐
一 乐1 小时前
家政服务管理系统|基于springboot + vue家政服务管理系统(源码+数据库+文档)
java·数据库·vue.js·spring boot·论文·毕设·家政服务管理系统
碳基硅坊2 小时前
Spring AI:把大模型接进 Spring 应用
java·人工智能·spring ai
黄毛火烧雪下2 小时前
Java 核心知识点总结(一)
java·开发语言
技术小结-李爽3 小时前
【工具】Maven的下载、安装、使用
java·maven
极创信息3 小时前
Linux挖矿病毒深度清理实战教程,从进程隐藏、Rootkit驻留到彻底根除
java·大数据·linux·运维·安全·tomcat·健康医疗
努力成为AK大王3 小时前
并发编程的核心挑战、优化方案与核心知识点总结
java·开发语言·数据库
云烟成雨TD3 小时前
Agent Scope Java 2.x 系列【10】技能(Skill)
java·人工智能·agent
摇滚侠3 小时前
SpringMVC 入门到实战 DispatcherServlet 源码解读 92-95
java·后端·spring·maven·intellij-idea
键盘歌唱家3 小时前
Spring AI 入门分享:它和“直接调 API“到底差在哪
java·人工智能·spring
宸丶一4 小时前
Day 10:LangGraph - Agent 的图执行引擎
java·windows·python