Java异步编程利器:CompletableFuture完全指南



图:传统同步 vs 异步编程模式对比


一、为什么需要异步编程?

同步 vs 异步的餐厅点餐比喻

同步方式 异步方式
顾客站在柜台前等待厨师做完餐 顾客拿到取餐号后可以坐下玩手机
期间不能做其他事情 餐准备好后会通知顾客
类似传统Future.get()的阻塞调用 类似CompletableFuture的回调机制

现实场景

当你的应用需要:

  • 调用多个外部API
  • 执行数据库查询
  • 处理文件I/O

    使用异步编程可以显著提升系统吞吐量!

二、CompletableFuture 初体验

1. 创建最简单的异步任务

CompletableFuture.runAsync(() -> {
System.out.println(" 厨师开始烹饪...");
sleep(2000); // 模拟耗时操作
System.out.println(" 菜品完成!");
});
System.out.println(" 您可以继续玩手机等待通知");

输出顺序

 您可以继续玩手机等待通知
厨师开始烹饪...
(2秒后)
菜品完成!

2. 带返回值的异步任务

CompletableFuture<String> mealFuture = CompletableFuture.supplyAsync(() -> {
sleep(1500);
return " 芝士汉堡";
}); // 获取结果(阻塞方式,实际慎用)
String meal = mealFuture.join();
System.out.println("您点的 " + meal + " 已送达");

三、核心功能详解

1. 回调方法三剑客

方法 描述 示例
thenApply 转换结果 .thenApply(meal -> "热的 " + meal)
thenAccept 消费结果 .thenAccept(System.out::println)
thenRun 执行后续操作 .thenRun(() -> cleanKitchen())

示例代码

CompletableFuture.supplyAsync(() -> " 拉面")
.thenApply(dish -> " 热的 " + dish)
.thenAccept(System.out::println)
.thenRun(() -> System.out.println(" 打包完成"));

2. 任务组合技巧

(1) 顺序执行(汉堡+薯条套餐)

CompletableFuture.supplyAsync(() -> " 汉堡")
.thenCompose(burger ->
CompletableFuture.supplyAsync(() -> burger + " + 薯条"))
.thenAccept(System.out::println);

(2) 并行执行(同时准备主食和饮料)

CompletableFuture<String> mainCourse =
supplyAsync(() -> " 咖喱饭");
CompletableFuture<String> drink =
supplyAsync(() -> " 可乐"); mainCourse.thenCombine(drink, (food, drink) ->
"套餐:" + food + " 配 " + drink)
.thenAccept(System.out::println);

四、异常处理实战

1. 基础异常捕获

CompletableFuture.supplyAsync(() -> {
if (Math.random() > 0.5) {
throw new RuntimeException(" 厨房着火啦!");
}
return " 披萨";
}).exceptionally(ex -> {
System.out.println("补偿方案:" + ex.getMessage());
return " 免费面包";
});

2. 统一异常处理

.handle((result, ex) -> {
if (ex != null) {
return "️ 出餐失败:" + ex.getMessage();
}
return result + " (特制版)";
});

五、完整餐厅模拟系统

public class RestaurantSystem {
public static void main(String[] args) {
// 并行准备主菜和汤
CompletableFuture<String> mainCourse = cookAsync(" 牛排", 3000);
CompletableFuture<String> soup = cookAsync(" 蘑菇汤", 2000); // 组合结果
mainCourse.thenCombine(soup, (m, s) -> "主餐:" + m + "\n汤品:" + s)
.thenAccept(order -> {
System.out.println("====== 您的订单 ======");
System.out.println(order);
}); // 饭后甜点(顺序执行)
mainCourse.thenCompose(food ->
cookAsync(" 冰淇淋", 1000))
.thenAccept(dessert ->
System.out.println("餐后甜点:" + dessert));
} static CompletableFuture<String> cookAsync(String dish, int time) {
return CompletableFuture.supplyAsync(() -> {
System.out.println("‍ 开始制作:" + dish);
sleep(time);
return dish + " (已完成)";
});
}
}

六、学习路线图

  1. 新手阶段

    掌握 supplyAsync + thenAccept 基础组合

    理解异步编程思想

  2. 进阶阶段

    熟练使用 thenApply 数据转换

    学习 thenCompose 扁平化处理

  3. 高手阶段

    掌握多任务组合(allOf/anyOf

    深入理解异常处理机制

  4. 专家技巧

    自定义线程池优化性能

    与Stream API结合使用


关键思考

CompletableFuture 就像餐厅的点餐系统,让:

  • 厨师(工作线程)在后台烹饪
  • 服务员(回调方法)负责通知
  • 顾客(主线程)无需苦苦等待

掌握这个模式,你的Java应用性能将获得质的飞跃!

【Java异步编程利器】CompletableFuture完全指南的更多相关文章

  1. java异步编程降低延迟

    目录 java异步编程降低延迟 一.ExecutorService和CompletionService 二.CompletableFuture(重要) 三.stream中的parallel(并行流) ...

  2. 【Android实战】----从Retrofit源代码分析到Java网络编程以及HTTP权威指南想到的

    一.简单介绍 接上一篇[Android实战]----基于Retrofit实现多图片/文件.图文上传中曾说非常想搞明确为什么Retrofit那么屌. 近期也看了一些其源代码分析的文章以及亲自查看了源代码 ...

  3. [置顶] 【Android实战】----从Retrofit源码分析到Java网络编程以及HTTP权威指南想到的

    一.简介 接上一篇[Android实战]----基于Retrofit实现多图片/文件.图文上传中曾说非常想搞明白为什么Retrofit那么屌.最近也看了一些其源码分析的文章以及亲自查看了源码,发现其对 ...

  4. Java 异步编程的几种方式

    前言 异步编程是让程序并发运行的一种手段.它允许多个事情同时发生,当程序调用需要长时间运行的方法时,它不会阻塞当前的执行流程,程序可以继续运行,当方法执行完成时通知给主线程根据需要获取其执行结果或者失 ...

  5. Java 异步编程 (5 种异步实现方式详解)

    ​ 同步操作如果遇到一个耗时的方法,需要阻塞等待,那么我们有没有办法解决呢?让它异步执行,下面我会详解异步及实现@mikechen 目录 什么是异步? 一.线程异步 二.Future异步 三.Comp ...

  6. Paip.Php Java 异步编程。推模型与拉模型。响应式(Reactive)”编程FutureData总结... 1

    Paip.Php  Java 异步编程.推模型与拉模型.响应式(Reactive)"编程FutureData总结... 1.1.1       异步调用的实现以及角色(:调用者 提货单) F ...

  7. 异步编程利器:CompletableFuture

    一.一个示例回顾Future 一些业务场景我们需要使用多线程异步执行任务,加快任务执行速度. JDK5新增了Future接口,用于描述一个异步计算的结果.虽然 Future 以及相关使用方法提供了异步 ...

  8. Java 异步编程

    昨天头儿给的学习文档我还没看完,头儿说:“MongoDB光会简单的添删改查什么的不行,要深入了解,你们连$set和$inc使用场景都分不清.” 确实,学习过一年多SQL,确实对学习MongoDB有点影 ...

  9. Java异步编程——深入源码分析FutureTask

    Java的异步编程是一项非常常用的多线程技术. 之前通过源码详细分析了ThreadPoolExecutor<你真的懂ThreadPoolExecutor线程池技术吗?看了源码你会有全新的认识&g ...

  10. java异步编程

    异步编程提供了一个非阻塞事件驱动的模型.通过异步消除阻塞,可以让web服务响应更多请求.可以让系统更高效的执行.比如log框架,记录日志或异常时异步执行可避免影响正常业务流程的执行. 异步变成如何把线 ...

随机推荐

  1. Java技术栈总结-基础

    - - -计算机技术演化- - -1 编程语言演化1.1 写在最前  此文用于个人总结,串接知识点 1.2 汇编  举例:mov .add  特点:程序量很大,几百行.几千行乃至几万行 1.3 VB- ...

  2. w3cschool-Flink 入门

    Flink 入门   Apache Flink是一个框架和分布式处理引擎,用于在无界和有界数据流上进行有状态的计算.Flink被设计为在所有常见的集群环境中运行,以内存中的速度和任何规模执行计算. A ...

  3. 微信小程序block的作用

    有了block标签过后,你就可以把if 或则 for 语句写在block标签里面; 这样就控制了这一块的逻辑. 个人建议是要是v-if和v-for的都可以写在block上: block并不是一个组件, ...

  4. 手把手教你在本地部署DeepSeek R1,搭建web-ui ,建议收藏!

    写在前面 最近,DeepSeek 发布的推理大模型 DeepSeek - R1 ,可以说是AI大模型领域杀出的一匹黑马.它在国外大模型排名 Arena 上成绩惊人,基准测试位列全类别大模型第三,在风格 ...

  5. [记录点滴] 记录一次用 IntelliJ IDEA遇到scope provided 的坑

    0x00 问题 最近在调试一个网上的项目,结果遇到两个问题,特此记录下解决过程. 问题: 某一个jar包有版本冲突 某一个类,居然在IntelliJ IDEA中运行调试时候找不到 0x01 解决途径 ...

  6. jar包启停shell脚本

    jar包控制脚本1 #!/bin/bash export JAVA_HOME=/u01/java_home/jdk1.8.0_181 export APP_HOME=/u01/test export ...

  7. Atcoder ABC387F Count Arrays 题解 [ 绿 ] [ 基环树 ] [ 树形 dp ] [ 前缀和优化 ]

    Count Arrays:一眼秒的计数题. 思路 显然,把小于等于的条件化为大的向小的连单向边,每个数的入度都是 \(1\),就会形成一个基环树森林. 那么考虑这个环上能填什么数.因为所有数都小于等于 ...

  8. MOS管选型

    MOS管基本参数 MOS管(Metal-Oxide-Semiconductor Field-Effect Transistor, MOSFET)作为开关元件的应用非常广泛,其开关特性与三极管相比有所不 ...

  9. Hadoop - 两个Namenode都是standby状态怎么处理

    在任意一个standby的NN节点执行 再次访问 ctos01:9870页面

  10. Flink学习(二) 应用场景和架构模型

    实时计算最好的时代 在过去的十年里,面向数据时代的实时计算技术接踵而至.从我们最初认识的 Storm,再到 Spark 的异军突起,迅速占领了整个实时计算领域.直到 2019 年 1 月底,阿里巴巴内 ...