【Java异步编程利器】CompletableFuture完全指南
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 + " (已完成)";
});
}
}
六、学习路线图
新手阶段
掌握supplyAsync+thenAccept基础组合
理解异步编程思想进阶阶段
熟练使用thenApply数据转换
学习thenCompose扁平化处理高手阶段
掌握多任务组合(allOf/anyOf)
深入理解异常处理机制专家技巧
自定义线程池优化性能
与Stream API结合使用
关键思考:
CompletableFuture 就像餐厅的点餐系统,让:
- 厨师(工作线程)在后台烹饪
- 服务员(回调方法)负责通知
- 顾客(主线程)无需苦苦等待
掌握这个模式,你的Java应用性能将获得质的飞跃!
【Java异步编程利器】CompletableFuture完全指南的更多相关文章
- java异步编程降低延迟
目录 java异步编程降低延迟 一.ExecutorService和CompletionService 二.CompletableFuture(重要) 三.stream中的parallel(并行流) ...
- 【Android实战】----从Retrofit源代码分析到Java网络编程以及HTTP权威指南想到的
一.简单介绍 接上一篇[Android实战]----基于Retrofit实现多图片/文件.图文上传中曾说非常想搞明确为什么Retrofit那么屌. 近期也看了一些其源代码分析的文章以及亲自查看了源代码 ...
- [置顶]
【Android实战】----从Retrofit源码分析到Java网络编程以及HTTP权威指南想到的
一.简介 接上一篇[Android实战]----基于Retrofit实现多图片/文件.图文上传中曾说非常想搞明白为什么Retrofit那么屌.最近也看了一些其源码分析的文章以及亲自查看了源码,发现其对 ...
- Java 异步编程的几种方式
前言 异步编程是让程序并发运行的一种手段.它允许多个事情同时发生,当程序调用需要长时间运行的方法时,它不会阻塞当前的执行流程,程序可以继续运行,当方法执行完成时通知给主线程根据需要获取其执行结果或者失 ...
- Java 异步编程 (5 种异步实现方式详解)
同步操作如果遇到一个耗时的方法,需要阻塞等待,那么我们有没有办法解决呢?让它异步执行,下面我会详解异步及实现@mikechen 目录 什么是异步? 一.线程异步 二.Future异步 三.Comp ...
- Paip.Php Java 异步编程。推模型与拉模型。响应式(Reactive)”编程FutureData总结... 1
Paip.Php Java 异步编程.推模型与拉模型.响应式(Reactive)"编程FutureData总结... 1.1.1 异步调用的实现以及角色(:调用者 提货单) F ...
- 异步编程利器:CompletableFuture
一.一个示例回顾Future 一些业务场景我们需要使用多线程异步执行任务,加快任务执行速度. JDK5新增了Future接口,用于描述一个异步计算的结果.虽然 Future 以及相关使用方法提供了异步 ...
- Java 异步编程
昨天头儿给的学习文档我还没看完,头儿说:“MongoDB光会简单的添删改查什么的不行,要深入了解,你们连$set和$inc使用场景都分不清.” 确实,学习过一年多SQL,确实对学习MongoDB有点影 ...
- Java异步编程——深入源码分析FutureTask
Java的异步编程是一项非常常用的多线程技术. 之前通过源码详细分析了ThreadPoolExecutor<你真的懂ThreadPoolExecutor线程池技术吗?看了源码你会有全新的认识&g ...
- java异步编程
异步编程提供了一个非阻塞事件驱动的模型.通过异步消除阻塞,可以让web服务响应更多请求.可以让系统更高效的执行.比如log框架,记录日志或异常时异步执行可避免影响正常业务流程的执行. 异步变成如何把线 ...
随机推荐
- Java技术栈总结-基础
- - -计算机技术演化- - -1 编程语言演化1.1 写在最前 此文用于个人总结,串接知识点 1.2 汇编 举例:mov .add 特点:程序量很大,几百行.几千行乃至几万行 1.3 VB- ...
- w3cschool-Flink 入门
Flink 入门 Apache Flink是一个框架和分布式处理引擎,用于在无界和有界数据流上进行有状态的计算.Flink被设计为在所有常见的集群环境中运行,以内存中的速度和任何规模执行计算. A ...
- 微信小程序block的作用
有了block标签过后,你就可以把if 或则 for 语句写在block标签里面; 这样就控制了这一块的逻辑. 个人建议是要是v-if和v-for的都可以写在block上: block并不是一个组件, ...
- 手把手教你在本地部署DeepSeek R1,搭建web-ui ,建议收藏!
写在前面 最近,DeepSeek 发布的推理大模型 DeepSeek - R1 ,可以说是AI大模型领域杀出的一匹黑马.它在国外大模型排名 Arena 上成绩惊人,基准测试位列全类别大模型第三,在风格 ...
- [记录点滴] 记录一次用 IntelliJ IDEA遇到scope provided 的坑
0x00 问题 最近在调试一个网上的项目,结果遇到两个问题,特此记录下解决过程. 问题: 某一个jar包有版本冲突 某一个类,居然在IntelliJ IDEA中运行调试时候找不到 0x01 解决途径 ...
- jar包启停shell脚本
jar包控制脚本1 #!/bin/bash export JAVA_HOME=/u01/java_home/jdk1.8.0_181 export APP_HOME=/u01/test export ...
- Atcoder ABC387F Count Arrays 题解 [ 绿 ] [ 基环树 ] [ 树形 dp ] [ 前缀和优化 ]
Count Arrays:一眼秒的计数题. 思路 显然,把小于等于的条件化为大的向小的连单向边,每个数的入度都是 \(1\),就会形成一个基环树森林. 那么考虑这个环上能填什么数.因为所有数都小于等于 ...
- MOS管选型
MOS管基本参数 MOS管(Metal-Oxide-Semiconductor Field-Effect Transistor, MOSFET)作为开关元件的应用非常广泛,其开关特性与三极管相比有所不 ...
- Hadoop - 两个Namenode都是standby状态怎么处理
在任意一个standby的NN节点执行 再次访问 ctos01:9870页面
- Flink学习(二) 应用场景和架构模型
实时计算最好的时代 在过去的十年里,面向数据时代的实时计算技术接踵而至.从我们最初认识的 Storm,再到 Spark 的异军突起,迅速占领了整个实时计算领域.直到 2019 年 1 月底,阿里巴巴内 ...