Java CompletableFuture:你真的了解它吗?

news2024/11/24 3:03:56

在这里插入图片描述

文章目录

    • 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 -> {
    // 使用第一个任务的结果来创建新的 CompletableFuture
    return CompletableFuture.supplyAsync(() -> result1 + " 处理完成");
});

对于这些方法来说,它们为处理多个异步任务提供了强大的工具,使得并发编程更加高效和灵活。通过合理使用这些方法,可以实现复杂的异步任务组合和处理逻辑,确保程序的高效执行。

世界会向那些有目标和远见的人让路

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2064044.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

springboot静态资源访问问题归纳

以下内容基于springboot 2.3.4.RELEASE 1、默认配置的springboot项目&#xff0c;有四个静态资源文件夹&#xff0c;它们是有优先级的&#xff0c;如下&#xff1a; "classpath:/META-INF/resources/", &#xff08;优先级最高&#xff09; "classpath:/reso…

【精选】基于Spark的国漫推荐系统(精选设计产品)

目录&#xff1a; 系统开发技术 Python可视化技术 Django框架 Hadoop介绍 Scrapy介绍 IDEA介绍 B/S架构 MySQL数据库介绍 系统流程分析 操作流程 添加信息流程 删除信息流程 系统系统介绍&#xff1a; 可以查看我的B站&#xff1a; 系统测试 运行环境 软件平台 硬…

docker-compose安装NebulaGraph 3.8.0

文章目录 一. 安装NebulaGraph1.1 通过 Git 克隆nebula-docker-compose仓库的3.8.0分支到主机1.2 部署1.3 卸载1.4 查看 二. 安装NebulaGraph Studio2.1 下载 Studio 的部署配置文件2.2 创建nebula-graph-studio-3.10.0目录&#xff0c;并将安装包解压至目录中2.3 解压后进入 n…

shaushaushau1

CVE-2023-7130 靶标介绍&#xff1a; College Notes Gallery 2.0 允许通过“/notes/login.php”中的参数‘user’进行 SQL 注入。利用这个问题可能会使攻击者有机会破坏应用程序&#xff0c;访问或修改数据. 已经告诉你在哪里存在sql注入了&#xff0c;一般上来应该先目录扫…

【补充篇】AUTOSAR多核OS介绍(下)

文章目录 前文回顾1 AUTOSAR OS1.1 AUTSOAR OS元素1.1.1 操作系统对象1.1.2 操作系统应用程序1.1.3 AUTOSAR OS裁剪类型1.1.4 AUTOSAR OS软件分区1.2 AUTOSAR OS自旋锁1.3 AUTOSAR OS核间通信1.4 AUTOSAR OS多核调度前文回顾 在上篇文章【补充篇】AUTOSAR多核OS介绍(上)中,…

对于一个36岁的人来说,现在转行AI大模型还来得及吗?

前言 在职场生涯中&#xff0c;33岁似乎是一个尴尬的年龄。许多人在这个阶段已经定型&#xff0c;难以寻求新的突破。然而&#xff0c;随着科技行业的飞速发展&#xff0c;人工智能成为了新时代的宠儿。那么&#xff0c;对于一个33岁的人来说&#xff0c;现在转行AI大模型还来…

做SSH实验下载 paramiko库

今天做SSH实验下载paramiko库文件一直出问题&#xff0c;后面库文件下好了还是报错&#xff0c;这里记录了我的解决方案。 pycharm修改默认下载路径为国内镜像&#xff08;我这里用清华大学的镜像下载快一些&#xff09; Simple Index 到这里路径就改好了&#xff0c;接下来就…

从就业出发,深度剖析大数据行业的现状与前景

以一个经典案例引入——啤酒与纸尿裤的故事。 20世纪90年代&#xff0c;沃尔玛从购物的后台信息数据中&#xff0c;发现很多买了纸尿裤的男士会同时买啤酒。后来&#xff0c;调查发现&#xff0c;此类人多是被“轰出来”买纸尿裤&#xff0c;一想到养娃压力大&#xff0c;心情…

牛客竞赛数据结构专题班树状数组、线段树练习题

牛客竞赛_ACM/NOI/CSP/CCPC/ICPC算法编程高难度练习赛_牛客竞赛OJ G 智乃酱的平方数列&#xff08;线段树&#xff0c;等差数列&#xff0c;多项式&#xff09; 题目描述 想必你一定会用线段树维护等差数列吧&#xff1f;让我们来看看它的升级版。 请你维护一个长度为510 ^5…

Mysql高级 [Linux版] 性能优化 数据库系统配置优化 和 MySQL的执行顺序 以及 Mysql执行引擎介绍

数据库系统配置优化 1、定义 数据库是基于操作系统的&#xff0c;目前大多数MySQL都是安装在linux系统之上&#xff0c;所以对于操作系统的一些参数配置也会影响到MySQL的性能&#xff0c;下面就列出一些常用的系统配置。 2、优化配置参数-操作系统 优化包括操作系统的优化及My…

集运系统:如何实现不同员工的不同操作权限?

在集运行业&#xff0c;员工的角色和职责各有不同&#xff0c;因此对系统的操作权限需求也不尽相同。为了确保数据的安全性和业务的顺利进行&#xff0c;易境通集运系统提供了灵活的权限管理功能&#xff0c;让企业可以根据员工的角色和职责&#xff0c;设置不同的操作权限。 易…

Redis (day 3)

一、通过jedis连接数据库 1.首先导入依赖 <!-- https://mvnrepository.com/artifact/redis.clients/jedis --><dependency><groupId>redis.clients</groupId><artifactId>jedis</artifactId><version>5.1.0</version></de…

mac 微信数据直接存储到移动硬盘

在apple设备上存储都是1500块/128gb的价格收取的&#xff0c;真的是寸土寸金。在手机已经占用了一遍存储空间之后&#xff0c;微信备份还要占用一遍。 iCloud备份微信聊天记录的稳定性真的非常差劲&#xff0c;比如我微信30g&#xff0c;经常恢复到20g左右就被打断&#xff0c;…

【C++ Primer Plus习题】2.6

问题: 解答: #include <iostream> using namespace std;#define LIGHT_TO_SKY 63240double lightToSky(double value) {return value * LIGHT_TO_SKY; }int main() {double light 0;cout << "请输入光年值:";cin >> light;cout << light &…

还在返回一大堆 null 字段给前端?

在许多情况下&#xff0c;返回的 JSON 数据可能包含许多 null 值的字段&#xff0c;这会导致数据冗余&#xff0c;增加网络传输的负担&#xff0c;并使得前端处理数据变得复杂。因此&#xff0c;使用 JsonInclude(JsonInclude.Include.NON_NULL) 可以帮助我们优化 JSON 的输出&…

看看人家写的,Controller太优雅了~【送源码】

今天咱们来聊聊如何写出优雅的Controller代码。 写程序想让作品成为经典&#xff0c;不只是简单地加个try-catch就完事了。有时候&#xff0c;一个不小心&#xff0c;Controller里写的业务逻辑都能让你血压飙升&#xff01;不过别慌&#xff0c;今天我就来带大家看看怎么把Cont…

文件描述符的复制,访问测试,修改文件大小,文件锁

指定文件描述符 #include <unistd.h>int dup2(int oldfd, int newfd); ->功能:复制文件描述符表的特定条目到指定项&#xff0c; ->参数:oldfd: 源文件描述符 newfd: 目标文件描述符 ->返回值:成功返回目标文件描述符(newfd)&#xff0c;失败返回…

【C++ 面试 - 面向对象】每日 3 题(八)

✍个人博客&#xff1a;Pandaconda-CSDN博客 &#x1f4e3;专栏地址&#xff1a;http://t.csdnimg.cn/fYaBd &#x1f4da;专栏简介&#xff1a;在这个专栏中&#xff0c;我将会分享 C 面试中常见的面试题给大家~ ❤️如果有收获的话&#xff0c;欢迎点赞&#x1f44d;收藏&…

P(查准率) R(查全率) AP mAP最通俗准确的讲解

学习YOLO的过程中遇到了mAP指标&#xff0c;在网上看了很多关于mAP的讲解&#xff0c;不是很理解其计算过程&#xff0c;于是总结了各个帖子及自己的理解&#xff0c;给出mAP计算的规律&#xff0c;这样就能很好的记忆。 目录 一、P&#xff08;精确率&#xff09;、R&#x…