Java CompletableFuture:你真的了解它吗?
文章目录
- 1 什么是 CompletableFuture?
- 2 如何正确使用 CompletableFuture 对象?
- 3 如何结合回调函数处理异步任务结果?
- 4 如何组合并处理多个 CompletableFuture?
1 什么是 CompletableFuture?
CompletableFuture 是 Java 8 引入的一个强大的异步编程工具。允许以声明式的方式处理异步任务的结果,避免了传统回调和手动管理线程的复杂性。
CompletableFuture 可以组合和链式调用,高效地利用多核处理器的能力,并且减少了传统并发编程中常见的竞态条件和死锁等问题。
在日常开发中,经常需要处理那些可能耗时的任务,比如网络请求、数据库查询或者复杂的计算。使用 CompletableFuture,可以告诉程序如何在后台执行这些任务,然后在任务完成后执行特定的操作。
可以想象一下,CompletableFuture 就像是一条可以穿越时间的信使,你可以把一项任务托付给它,然后继续做其他事情。当任务完成时,它会及时将结果送回来,让你可以立即处理。这样,你就不必在等待任务完成的过程中浪费时间,而是可以更高效地利用自己的资源。
2 如何正确使用 CompletableFuture 对象?
CompletableFuture 可以以一种非阻塞的方式执行异步任务,并能够在任务完成后立即得到通知。通过链式调用的方式,可以很方便地组合多个异步操作,处理它们的结果或者异常。
通过 CompletableFuture.supplyAsync()
方法创建一个 CompletableFuture 对象,并指定一个需要异步执行的任务:
// Supplier 函数会在一个新的线程上异步执行
CompletableFuture<String> future = CompletableFuture.supplyAsync(() -> {// 模拟一个耗时操作,如从数据库中读取数据try {Thread.sleep(2000);} catch (InterruptedException e) {e.printStackTrace();}return "异步任务完成";
});
通过在 CompletableFuture 上添加一些操作,比如处理任务的结果或者处理任务执行过程中可能发生的异常:
// thenAccept()方法接收一个 Consumer 函数
future.thenAccept(result -> {System.out.println("任务完成,结果为:" + result);
}).exceptionally(ex -> {System.out.println("任务出现异常:" + ex.getMessage());return null;
});
3 如何结合回调函数处理异步任务结果?
结合回调函数处理异步任务结果的过程可以比作在等待一份重要的快递时安排一个通知服务。这个通知服务就是回调函数,它会在快递送达时通知你,或者在处理完成后执行特定的操作。
在 Java 的 CompletableFuture
中,这种模式可以通过 supplyAsync()
、thenApply()
、thenAccept()
和 handle()
方法来实现。
创建一个异步任务时,使用 CompletableFuture.supplyAsync()
可以启动一个任务,这个任务在后台线程中执行,直到它完成。假设有一个任务需要从远程服务器获取数据:
// supplyAsync() 方法接收一个 Supplier 函数,这个函数会在后台线程中运行,并返回一个结果
// 结果会被封装在 CompletableFuture 对象中,等待进一步处理
CompletableFuture<String> future = CompletableFuture.supplyAsync(() -> {// 模拟从远程服务器获取数据try {Thread.sleep(3000);} catch (InterruptedException e) {e.printStackTrace();}return "数据已成功获取";
});
使用 thenApply()
方法,可以在异步任务完成后,对结果进行转换。这个方法接收一个 Function
函数,这个函数会接收任务的结果,并返回一个新结果。比如,将获取的数据进行处理:
// thenApply() 方法将原始数据转换为大写形式
// 处理后的结果会成为新的 CompletableFuture 对象的结果
CompletableFuture<String> processedFuture = future.thenApply(result -> {// 对结果进行处理return result.toUpperCase();
});
为了执行一个操作而不关心处理的结果,可以使用 thenAccept()
方法。这个方法接收一个 Consumer
函数,它处理任务完成时的结果,可以在异步任务完成时执行一些操作,比如日志记录或通知用户。例如,将结果打印到控制台:
// 在任务完成后会调用传入的 Consumer 函数,并将结果传递给它
future.thenAccept(result -> {System.out.println("任务完成,结果是:" + result);
});
在任务执行过程中,可能会遇到异常。handle()
方法可以用来处理这些异常,它接收一个 BiFunction
函数,这个函数接收结果和异常(如果有的话),并返回一个处理后的结果。例如:
// handle() 方法检查是否有异常发生
// 如果有异常,它会处理异常并返回一个默认的结果
// 如果没有异常,它会处理正常的结果
CompletableFuture<String> handledFuture = future.handle((result, ex) -> {if (ex != null) {// 处理异常System.out.println("任务发生错误:" + ex.getMessage());return "错误处理结果";}// 处理正常结果return result.toLowerCase();
});
对于这四种回调函数,可以使得异步任务的结果处理变得灵活而强大。通过结合使用不同的回调函数,可以对异步任务的结果进行多种操作,保证程序在处理复杂任务时仍然保持清晰和高效。
4 如何组合并处理多个 CompletableFuture?
组合和处理多个 CompletableFuture
可以让并发任务变得更加灵活和高效。设想有多个任务需要并行执行,然后将它们的结果结合起来进行进一步处理。
在进行组合时,最基本的方法之一是将多个 CompletableFuture
的结果合并。比如,有两个任务需要并行完成,获取两个不同的数据源,然后将这两个结果结合起来。
可以使用 thenCombine()
方法,它接收两个 CompletableFuture
和一个合并函数,两个 CompletableFuture
必须在相同的线程池中执行。
假设有两个任务分别从不同的 API 获取数据:
CompletableFuture<String> future1 = CompletableFuture.supplyAsync(() -> {// 模拟从第一个 API 获取数据return "数据1";
});CompletableFuture<String> future2 = CompletableFuture.supplyAsync(() -> {// 模拟从第二个 API 获取数据return "数据2";
});
为了将这两个结果结合起来,可以使用 thenCombine()
:
// thenCombine() 方法接收两个 CompletableFuture 和一个函数,这个函数将两个任务的结果合并成一个结果
// 最终的结果是将两个字符串连接在一起
CompletableFuture<String> combinedFuture = future1.thenCombine(future2, (result1, result2) -> {// 将两个结果结合成一个return result1 + " 和 " + result2;
});
另一个有用的方法是 allOf()
方法。当有多个任务需要并行执行,并且在所有任务完成后执行某个操作时,allOf()
非常有用。它接收一个 CompletableFuture
数组,并在所有这些 CompletableFuture
完成时触发。可以用来等待多个异步任务完成,然后执行某个操作:
CompletableFuture<Void> allOfFuture = CompletableFuture.allOf(future1, future2);
要获取所有任务的结果,可以在 allOf()
的结果上添加一个回调函数:
// thenRun() 方法会在所有任务完成后执行,它不需要处理结果,只是执行某个操作
allOfFuture.thenRun(() -> {// 处理所有任务完成后的操作try {String result1 = future1.get();String result2 = future2.get();System.out.println("任务1的结果: " + result1);System.out.println("任务2的结果: " + result2);} catch (InterruptedException | ExecutionException e) {e.printStackTrace();}
});
如果有任务依赖于另一个任务的结果,可以使用 thenCompose()
方法。这种情况下,第二个任务会在第一个任务完成后开始执行。thenCompose()
方法接收一个返回 CompletableFuture
的函数,然后将这两个 CompletableFuture
链接起来:
CompletableFuture<String> future3 = future1.thenCompose(result1 -> {// 使用第一个任务的结果来创建新的 CompletableFuturereturn CompletableFuture.supplyAsync(() -> result1 + " 处理完成");
});
对于这些方法来说,它们为处理多个异步任务提供了强大的工具,使得并发编程更加高效和灵活。通过合理使用这些方法,可以实现复杂的异步任务组合和处理逻辑,确保程序的高效执行。
世界会向那些有目标和远见的人让路
相关文章:

Java CompletableFuture:你真的了解它吗?
文章目录 1 什么是 CompletableFuture?2 如何正确使用 CompletableFuture 对象?3 如何结合回调函数处理异步任务结果?4 如何组合并处理多个 CompletableFuture? 1 什么是 CompletableFuture? CompletableFuture 是 Ja…...

5个免费在线 AI 绘画网站推荐,附100+提示词!
在数字化时代,艺术创作与人工智能的结合已带来前所未有的创新体验。AI 绘画技术,基于先进的人工智能算法,为艺术创作提供了全新的视角和工具。当前,多个免费在线AI绘画平台应运而生,为创作者们提供了丰富的灵感和创作机…...
C++基础语法:while的使用
前言 "打牢基础,万事不愁" .C的基础语法的学习."学以致用,边学边用",编程是实践性很强的技术,在运用中理解,总结. 引入 while的使用是编写代码的基础内容.笔者的记忆力已不如以前,最近遇到了还花了不少功夫,可见是掌握地不够牢固.所以对while的思路和内容…...

鹏哥C语言自定义笔记重点(29-)
29.函数指针数组 30.void指针是不能直接解引用,也不能-整数。 void*是无具体类型的指针,可以接受任何类型的地址。 31.qsort:使用快速排序的思想实现一个排序函数(升序) 32. 33.地址的字节是4/8 34.char arr[]{a,b} sizeof(arr[0]1)答案是4࿰…...
代码随想录算法训练营第六十天 | dijkstra(堆优化版)、Bellman_ford 算法精讲
一、dijkstra(堆优化版) 题目连接:47. 参加科学大会(第六期模拟笔试) (kamacoder.com) 文章讲解:代码随想录 (programmercarl.com)——dijkstra(堆优化版) 二、Bellman_ford 算法精讲…...
boost::asio 库版本,C/C++代码编译兼容性
1、boost::asio::spawn 开启有栈(stackful)协同程序,版本改进及限制 > boost_1_80 版本应采用以下方式。 auto f [self, this](const boost::asio::yield_context& y) noexcept {bool success_ do_handshake(y);if (!success_) {clo…...
前端开发的项目导入方法与应用
前端项目启动问题归集: 由于前端的项目对于npm的版本有要求,需要将其升级到20,所以必要的时候通过nvm,或者直接下载最新的安装包进行npm覆盖安装。在项目目录中应用npm i安装node_modules,如果没有正常安装的话&#…...

C++:模拟实现string
前言: 为了更好的理解string底层的原理,我们将模拟实现string类中常用的函数接口。为了与std里的string进行区分,所以用命名空间来封装一个自己的strin类。 string.h #pragma once #define _CRT_SECURE_NO_WARNINGS 1#include<iostream&…...
浅谈Kafka(一)
浅谈Kafka(一) 文章目录 浅谈Kafka(一)Kafa的设计是什么样的数据传输的事务定义消息队列的应用场景Kafka怎么样判断节点是否存活Kafka的消息是采用pull模式还是push模式Kafka在磁盘上的消息格式Kafka高效文件存储设计特点Kafka与传…...

Redis7基础篇(八)
redis集群 是什么 能干吗 集群算法-分片-槽位slot redis集群的槽位slot redis集群的分片 分片和槽位的优势 槽位映射的解决方案 上面的三个方案分别对应了小厂 中厂 大厂 哈希槽取余分区 缺点 一致性哈希算法分区 小总结 哈希槽分区 经典面试题 这里说的redis是ap而不是cp的 …...
Tauri简介
在Tauri应用中,Rust和前端(通常是基于Web技术如React、Vue或Angular)之间的交互是一个核心特性,它允许开发者利用Rust的强大功能和性能,同时保持前端开发的灵活性和丰富的生态系统。这种交互主要通过Tauri提供的API桥接…...

JavaWeb——MVC架构模式
一、概述: MVC(Model View Controller)是软件工程中的一种 软件架构模式 ,它把软件系统分为模型、视图和控制器三个基本部分。用一种业务逻辑、数据、界面显示分离的方法组织代码,将业务逻辑聚集到一个部件里面,在改进和个性化定制界面及用户…...

Excel求和方法之
一 SUM(),选择要相加的数,回车即可 二 上面的方法还不够快。用下面这个 就成功了 三 还有一种一样快的 选中之后,按下Alt键和键(即Alt)...

Windows Server 域控制服务器安装及相关使用
目录 1.将客户机加入域 2.安装域控制器 3.新建域用户 4.设置用户登录时间,账户过期时间 5.软件分发 编辑 6.换壁纸 7.OU与GPO的概念 域为集中控制,拿下域控是拿下目标的关键 以Windows Server 2022为例 1.将客户机加入域 前提:客…...
linux基础命令(超级详细)
Linux 系统提供了丰富的命令行工具,用于各种文件操作、系统管理和网络配置等任务。以下是一些常用的 Linux 基础命令: 一、 文件和目录操作 1. ls: 列出目录内容 ls 列出当前目录的文件和目录 ls -l 以长格式列出文件和目录,包…...

大模型笔记之-XTuner微调个人小助手认知
前言 使用XTuner 微调个人小助手认知 一、下载模型 #安装魔搭依赖包 pip install modelscope新建download.py内容如下 其中Shanghai_AI_Laboratory/internlm2-chat-1_8b是魔搭对应的模型ID cache_dir/home/aistudio/data/model’为指定下载到本地的目录 from modelscope im…...

用TensorFlow实现线性回归
说明 本文采用TensorFlow框架进行讲解,虽然之前的文章都采用mxnet,但是我发现tensorflow提供了免费的gpu可供使用,所以果断开始改为tensorflow,若要实现文章代码,可以使用colaboratory进行运行,当然&#…...

IT计算机软件系统类毕业论文结构指南:从标题到结论的全景视角
一、背景 在快速发展的IT和人工智能领域,毕业论文不仅是学术研究的重要成果,也展示了学生掌握新技术和应用的能力。随着大数据和智能系统的复杂性增加,毕业设计(毕设)的论文章节安排变得尤为关键。一个结构清晰、内容详…...
leetcode27:移除元素(正解)
移除元素 给你一个数组 nums 和一个值 val,你需要 原地 移除所有数值等于 val 的元素。元素的顺序可能发生改变。然后返回 nums 中与 val 不同的元素的数量。 假设 nums 中不等于 val 的元素数量为 k,要通过此题,您需要执行以下操作…...
docker部署nginx--(部署静态文件和服务)
文档参考 1、http://testingpai.com/article/1649671014266 2、下载nginx docker pull nginx:alpine 然后启动nginx, docker run --rm -it -p 9192:80 nginx:alpine /bin/sh 关闭容器后,自动删除该容器 进入后,启动nginx, nginx进行curl h…...

centos 7 部署awstats 网站访问检测
一、基础环境准备(两种安装方式都要做) bash # 安装必要依赖 yum install -y httpd perl mod_perl perl-Time-HiRes perl-DateTime systemctl enable httpd # 设置 Apache 开机自启 systemctl start httpd # 启动 Apache二、安装 AWStats࿰…...

SiFli 52把Imagie图片,Font字体资源放在指定位置,编译成指定img.bin和font.bin的问题
分区配置 (ptab.json) img 属性介绍: img 属性指定分区存放的 image 名称,指定的 image 名称必须是当前工程生成的 binary 。 如果 binary 有多个文件,则以 proj_name:binary_name 格式指定文件名, proj_name 为工程 名&…...

[大语言模型]在个人电脑上部署ollama 并进行管理,最后配置AI程序开发助手.
ollama官网: 下载 https://ollama.com/ 安装 查看可以使用的模型 https://ollama.com/search 例如 https://ollama.com/library/deepseek-r1/tags # deepseek-r1:7bollama pull deepseek-r1:7b改token数量为409622 16384 ollama命令说明 ollama serve #:…...
解决:Android studio 编译后报错\app\src\main\cpp\CMakeLists.txt‘ to exist
现象: android studio报错: [CXX1409] D:\GitLab\xxxxx\app.cxx\Debug\3f3w4y1i\arm64-v8a\android_gradle_build.json : expected buildFiles file ‘D:\GitLab\xxxxx\app\src\main\cpp\CMakeLists.txt’ to exist 解决: 不要动CMakeLists.…...
32单片机——基本定时器
STM32F103有众多的定时器,其中包括2个基本定时器(TIM6和TIM7)、4个通用定时器(TIM2~TIM5)、2个高级控制定时器(TIM1和TIM8),这些定时器彼此完全独立,不共享任何资源 1、定…...

WebRTC调研
WebRTC是什么,为什么,如何使用 WebRTC有什么优势 WebRTC Architecture Amazon KVS WebRTC 其它厂商WebRTC 海康门禁WebRTC 海康门禁其他界面整理 威视通WebRTC 局域网 Google浏览器 Microsoft Edge 公网 RTSP RTMP NVR ONVIF SIP SRT WebRTC协…...

CTF show 数学不及格
拿到题目先查一下壳,看一下信息 发现是一个ELF文件,64位的 用IDA Pro 64 打开这个文件 然后点击F5进行伪代码转换 可以看到有五个if判断,第一个argc ! 5这个判断并没有起太大作用,主要是下面四个if判断 根据题目…...
2025.6.9总结(利与弊)
凡事都有两面性。在大厂上班也不例外。今天找开发定位问题,从一个接口人不断溯源到另一个 接口人。有时候,不知道是谁的责任填。将工作内容分的很细,每个人负责其中的一小块。我清楚的意识到,自己就是个可以随时替换的螺丝钉&…...

【阅读笔记】MemOS: 大语言模型内存增强生成操作系统
核心速览 研究背景 研究问题:这篇文章要解决的问题是当前大型语言模型(LLMs)在处理内存方面的局限性。LLMs虽然在语言感知和生成方面表现出色,但缺乏统一的、结构化的内存架构。现有的方法如检索增强生成(RA…...
基于Uniapp的HarmonyOS 5.0体育应用开发攻略
一、技术架构设计 1.混合开发框架选型 (1)使用Uniapp 3.8版本支持ArkTS编译 (2)通过uni-harmony插件调用原生能力 (3)分层架构设计: graph TDA[UI层] -->|Vue语法| B(Uniapp框架)B --&g…...