如何在Java中使用CompletableFuture处理异常

CompletableFuture通过exceptionally、handle和whenComplete方法实现异步异常处理:exceptionally仅在失败时提供备用结果,handle统一处理成功与失败并可转换结果,whenComplete则用于执行日志等副作用而不改变状态。

如何在java中使用completablefuture处理异常

在Java中,

CompletableFuture

处理异步操作的异常,核心在于它提供了一系列方法来拦截、转换或响应任务执行过程中出现的错误。你可以把它想象成一个异步的

try-catch

机制,让你能够在未来的某个时间点,优雅地处理那些可能发生的意料之外。

解决方案

CompletableFuture

提供了几种主要机制来处理异常,它们各有侧重,但目标都是为了让你能够控制异步操作的失败路径。

exceptionally(Function fn)

: 这个方法就像同步代码中的

catch

块。如果前一个

CompletableFuture

异常完成,那么

fn

会被调用,接收到异常作为参数。

fn

的返回值将成为这个新的

CompletableFuture

的成功结果。如果原始

CompletableFuture

成功完成,

exceptionally

就会被跳过。它主要用于提供一个备用值或默认值,将失败的

CompletableFuture

转换为一个成功的。

handle(BiFunction fn)

: 这个方法则更像一个结合了

try-catch-finally

的操作。

fn

总是会被调用,无论前一个

CompletableFuture

是成功还是失败。如果成功,

fn

会收到结果

T

和一个

null

Throwable

;如果失败,它会收到一个

null

的结果和实际的

Throwable

。这让你有机会检查结果或异常,然后返回一个新的值,从而决定后续

CompletableFuture

的状态。

立即学习“Java免费学习笔记(深入)”;

whenComplete(BiConsumer action)

: 这个方法主要用于执行副作用操作,比如日志记录或资源清理。

action

也会总是被调用,接收结果和异常。但与

exceptionally

handle

不同的是,

whenComplete

不会改变它所附加的

CompletableFuture

的结果或异常状态。如果原始

CompletableFuture

失败了,它在执行完

whenComplete

后仍然会以失败状态向下传递,除非后续有

exceptionally

handle

进行处理。

在链中显式返回失败的

CompletableFuture

: 有时,你可能在

thenApply

thenCompose

这样的转换方法中检测到错误,并希望立即终止后续链条并抛出异常。这时,你可以返回

CompletableFuture.failedFuture(new Exception("..."))

以下是一些代码示例,展示了这些方法的实际应用:

import java.util.concurrent.CompletableFuture;import java.util.concurrent.ThreadLocalRandom;import java.util.concurrent.TimeUnit;public class CompletableFutureExceptionHandling {    public static void main(String[] args) throws InterruptedException {        System.out.println("--- 示例 1: 使用 exceptionally() ---");        CompletableFuture future1 = CompletableFuture.supplyAsync(() -> {            if (ThreadLocalRandom.current().nextBoolean()) {                throw new RuntimeException("任务 1 出错了!"); // 模拟随机失败            }            return "任务 1 的结果";        }).exceptionally(ex -> {            System.err.println("exceptionally 捕获到异常 (任务 1): " + ex.getMessage());            return "任务 1 的备用结果"; // 提供一个备用值        });        future1.thenAccept(result -> System.out.println("任务 1 最终结果: " + result));        System.out.println("n--- 示例 2: 使用 handle() ---");        CompletableFuture future2 = CompletableFuture.supplyAsync(() -> {            if (ThreadLocalRandom.current().nextBoolean()) {                throw new IllegalStateException("任务 2 因状态问题失败!"); // 模拟随机失败            }            return "任务 2 的结果";        }).handle((result, ex) -> {            if (ex != null) {                System.err.println("handle 捕获到异常 (任务 2): " + ex.getMessage());                return "任务 2 的处理后备用结果"; // 将失败转换为成功,并提供信息            }            return result + " (成功处理)"; // 成功时也进行转换        });        future2.thenAccept(result -> System.out.println("任务 2 最终结果: " + result));        System.out.println("n--- 示例 3: 使用 whenComplete() 进行副作用操作 ---");        CompletableFuture future3 = CompletableFuture.supplyAsync(() -> {            if (ThreadLocalRandom.current().nextBoolean()) {                throw new ArithmeticException("任务 3 发生算术错误!"); // 模拟随机失败            }            return "任务 3 的结果";        }).whenComplete((result, ex) -> {            if (ex != null) {                System.err.println("副作用: 任务 3 失败,异常信息: " + ex.getMessage());            } else {                System.out.println("副作用: 任务 3 成功完成,结果: " + result);            }        }).exceptionally(ex -> { // whenComplete 不改变状态,如果需要处理失败,仍需 exceptionally 或 handle            System.err.println("whenComplete 后捕获异常 (任务 3): " + ex.getMessage());            return "任务 3 的最终备用结果";        });        future3.thenAccept(result -> System.out.println("任务 3 最终结果: " + result));        System.out.println("n--- 示例 4: 链式调用中传播失败 ---");        CompletableFuture future4 = CompletableFuture.supplyAsync(() -> {            System.out.println("开始执行任务 4");            if (true) { // 强制失败,用于演示                throw new RuntimeException("任务 4 强制失败");            }            return 10;        }).thenApply(data -> { // 如果上一步失败,这一步不会执行            System.out.println("任务 4 正在处理数据: " + data);            return data * 2;        }).exceptionally(ex -> {            System.err.println("任务 4 链中捕获异常: " + ex.getMessage());            return -1; // 提供一个错误值        });        future4.thenAccept(val -> System.out.println("任务 4 最终值: " + val));        // 等待所有异步任务完成,以便观察输出        try {            TimeUnit.SECONDS.sleep(2);        } catch (InterruptedException e) {            Thread.currentThread().interrupt();        }    }}

exceptionally()

handle()

whenComplete()

之间有什么关键区别

这三者是

CompletableFuture

异常处理的核心,但它们的设计目的和行为模式有着显著差异,理解这些区别对于写出健壮的异步代码至关重要。

exceptionally()

专注于异常恢复。它只在之前的

CompletableFuture

遇到异常时才会被触发。如果上游任务成功完成,

exceptionally()

会被完全跳过。它的主要作用是提供一个“Plan B”,当主路径失败时,能返回一个预设的、默认的或经过转换的成功值,从而将一个失败的

Future

变为一个成功的

Future

。我个人在需要为某个可能失败的操作提供一个明确的“兜底”值时,会优先考虑它。

易森网络企业版 易森网络企业版

如果您是新用户,请直接将本程序的所有文件上传在任一文件夹下,Rewrite 目录下放置了伪静态规则和筛选器,可将规则添加进IIS,即可正常使用,不用进行任何设置;(可修改图片等)默认的管理员用户名、密码和验证码都是:yeesen系统默认关闭,请上传后登陆后台点击“核心管理”里操作如下:进入“配置管理”中的&ld

易森网络企业版 0 查看详情 易森网络企业版

handle()

则更加灵活和通用。无论上游

CompletableFuture

是成功还是失败,

handle()

都会执行。它会同时接收到结果(如果成功)和异常(如果失败),其中一个会是

null

。这种设计允许你在这里统一处理成功和失败两种情况,你可以根据传入的参数来决定返回什么。你可以选择继续传播成功,将异常转换为一个成功的结果,甚至重新抛出一个新的异常。在我看来,

handle()

是一个强大的“决策点”,当你需要对异步操作的最终状态进行统一的判断和转换时,它非常有用。

whenComplete()

的目的在于副作用操作。它也总是会执行,无论成功或失败,同样接收结果和异常。但与前两者最根本的区别在于,

whenComplete()

不会改变它所依附的

CompletableFuture

的状态。这意味着,如果上游

Future

失败了,即使

whenComplete()

执行了它的逻辑(比如打印日志),这个

Future

仍然会以失败状态向下传递。我通常用

whenComplete()

来做一些“事后清理”或者“状态报告”的工作,比如记录日志、更新监控指标,而不需要影响整个异步链的执行结果。

简而言之:

exceptionally()

只处理失败,并提供一个成功的回退值

handle()

处理成功和失败,并转换结果

whenComplete()

处理成功和失败,但不改变结果,主要用于副作用。

如何避免在

CompletableFuture

链中丢失异常信息?

CompletableFuture

的世界里,异常的“丢失”是一个常见且令人头疼的问题,它往往导致调试变得异常困难。这种情况通常不是指异常真的消失了,而是指它被悄无声息地处理了,或者被转换成了一个不那么有用的形式,导致你无法追溯其根源。

要避免这种情况,我认为有几个关键点:

首先,务必记录原始异常。无论你使用

exceptionally()

还是

handle()

来捕获异常,都应该在处理逻辑中将捕获到的

Throwable

完整地记录下来,包括堆信息。仅仅返回一个默认值而不记录异常,就等于把问题“扫到地毯下面”,将来出了问题,你根本不知道发生了什么。例如,

exceptionally(ex -> { log.error("Async task failed", ex); return "default"; })

远比

exceptionally(ex -> "default")

要负责得多。

其次,谨慎选择异常处理策略。如果你在

exceptionally()

handle()

中返回

null

或者一个不具代表性的值,而下游代码又依赖于非

null

或特定类型的值,那么你只是把一个异步异常转换成了一个更隐蔽的

NullPointerException

或类型转换异常。这比直接看到原始异常更糟糕。如果无法提供有意义的恢复值,考虑重新抛出一个更具业务含义的异常,或者使用

CompletableFuture.failedFuture(new MyBusinessException("..."))

来明确地传播失败。

再者,理解

join()

get()

的行为。当你在

CompletableFuture

上调用

join()

get()

(通常在异步链的末端或需要阻塞等待结果时),如果

CompletableFuture

以异常方式完成,这两个方法会将其内部的

CompletionException

ExecutionException

重新抛出。这意味着,如果你在

以上就是如何在Java中使用CompletableFuture处理异常的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/743131.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月25日 16:21:36
下一篇 2025年11月25日 16:21:58

相关推荐

  • Uniapp 中如何不拉伸不裁剪地展示图片?

    灵活展示图片:如何不拉伸不裁剪 在界面设计中,常常需要以原尺寸展示用户上传的图片。本文将介绍一种在 uniapp 框架中实现该功能的简单方法。 对于不同尺寸的图片,可以采用以下处理方式: 极端宽高比:撑满屏幕宽度或高度,再等比缩放居中。非极端宽高比:居中显示,若能撑满则撑满。 然而,如果需要不拉伸不…

    2025年12月24日
    400
  • 如何让小说网站控制台显示乱码,同时网页内容正常显示?

    如何在不影响用户界面的情况下实现控制台乱码? 当在小说网站上下载小说时,大家可能会遇到一个问题:网站上的文本在网页内正常显示,但是在控制台中却是乱码。如何实现此类操作,从而在不影响用户界面(UI)的情况下保持控制台乱码呢? 答案在于使用自定义字体。网站可以通过在服务器端配置自定义字体,并通过在客户端…

    2025年12月24日
    800
  • 如何在地图上轻松创建气泡信息框?

    地图上气泡信息框的巧妙生成 地图上气泡信息框是一种常用的交互功能,它简便易用,能够为用户提供额外信息。本文将探讨如何借助地图库的功能轻松创建这一功能。 利用地图库的原生功能 大多数地图库,如高德地图,都提供了现成的信息窗体和右键菜单功能。这些功能可以通过以下途径实现: 高德地图 JS API 参考文…

    2025年12月24日
    400
  • 如何使用 scroll-behavior 属性实现元素scrollLeft变化时的平滑动画?

    如何实现元素scrollleft变化时的平滑动画效果? 在许多网页应用中,滚动容器的水平滚动条(scrollleft)需要频繁使用。为了让滚动动作更加自然,你希望给scrollleft的变化添加动画效果。 解决方案:scroll-behavior 属性 要实现scrollleft变化时的平滑动画效果…

    2025年12月24日
    000
  • 如何为滚动元素添加平滑过渡,使滚动条滑动时更自然流畅?

    给滚动元素平滑过渡 如何在滚动条属性(scrollleft)发生改变时为元素添加平滑的过渡效果? 解决方案:scroll-behavior 属性 为滚动容器设置 scroll-behavior 属性可以实现平滑滚动。 html 代码: click the button to slide right!…

    2025年12月24日
    500
  • 如何选择元素个数不固定的指定类名子元素?

    灵活选择元素个数不固定的指定类名子元素 在网页布局中,有时需要选择特定类名的子元素,但这些元素的数量并不固定。例如,下面这段 html 代码中,activebar 和 item 元素的数量均不固定: *n *n 如果需要选择第一个 item元素,可以使用 css 选择器 :nth-child()。该…

    2025年12月24日
    200
  • 使用 SVG 如何实现自定义宽度、间距和半径的虚线边框?

    使用 svg 实现自定义虚线边框 如何实现一个具有自定义宽度、间距和半径的虚线边框是一个常见的前端开发问题。传统的解决方案通常涉及使用 border-image 引入切片图片,但是这种方法存在引入外部资源、性能低下的缺点。 为了避免上述问题,可以使用 svg(可缩放矢量图形)来创建纯代码实现。一种方…

    2025年12月24日
    100
  • 如何让“元素跟随文本高度,而不是撑高父容器?

    如何让 元素跟随文本高度,而不是撑高父容器 在页面布局中,经常遇到父容器高度被子元素撑开的问题。在图例所示的案例中,父容器被较高的图片撑开,而文本的高度没有被考虑。本问答将提供纯css解决方案,让图片跟随文本高度,确保父容器的高度不会被图片影响。 解决方法 为了解决这个问题,需要将图片从文档流中脱离…

    2025年12月24日
    000
  • 为什么 CSS mask 属性未请求指定图片?

    解决 css mask 属性未请求图片的问题 在使用 css mask 属性时,指定了图片地址,但网络面板显示未请求获取该图片,这可能是由于浏览器兼容性问题造成的。 问题 如下代码所示: 立即学习“前端免费学习笔记(深入)”; icon [data-icon=”cloud”] { –icon-cl…

    2025年12月24日
    200
  • 如何利用 CSS 选中激活标签并影响相邻元素的样式?

    如何利用 css 选中激活标签并影响相邻元素? 为了实现激活标签影响相邻元素的样式需求,可以通过 :has 选择器来实现。以下是如何具体操作: 对于激活标签相邻后的元素,可以在 css 中使用以下代码进行设置: li:has(+li.active) { border-radius: 0 0 10px…

    2025年12月24日
    100
  • 如何模拟Windows 10 设置界面中的鼠标悬浮放大效果?

    win10设置界面的鼠标移动显示周边的样式(探照灯效果)的实现方式 在windows设置界面的鼠标悬浮效果中,光标周围会显示一个放大区域。在前端开发中,可以通过多种方式实现类似的效果。 使用css 使用css的transform和box-shadow属性。通过将transform: scale(1.…

    2025年12月24日
    200
  • 为什么我的 Safari 自定义样式表在百度页面上失效了?

    为什么在 Safari 中自定义样式表未能正常工作? 在 Safari 的偏好设置中设置自定义样式表后,您对其进行测试却发现效果不同。在您自己的网页中,样式有效,而在百度页面中却失效。 造成这种情况的原因是,第一个访问的项目使用了文件协议,可以访问本地目录中的图片文件。而第二个访问的百度使用了 ht…

    2025年12月24日
    000
  • 如何用前端实现 Windows 10 设置界面的鼠标移动探照灯效果?

    如何在前端实现 Windows 10 设置界面中的鼠标移动探照灯效果 想要在前端开发中实现 Windows 10 设置界面中类似的鼠标移动探照灯效果,可以通过以下途径: CSS 解决方案 DEMO 1: Windows 10 网格悬停效果:https://codepen.io/tr4553r7/pe…

    2025年12月24日
    000
  • 使用CSS mask属性指定图片URL时,为什么浏览器无法加载图片?

    css mask属性未能加载图片的解决方法 使用css mask属性指定图片url时,如示例中所示: mask: url(“https://api.iconify.design/mdi:apple-icloud.svg”) center / contain no-repeat; 但是,在网络面板中却…

    2025年12月24日
    000
  • 如何用CSS Paint API为网页元素添加时尚的斑马线边框?

    为元素添加时尚的斑马线边框 在网页设计中,有时我们需要添加时尚的边框来提升元素的视觉效果。其中,斑马线边框是一种既醒目又别致的设计元素。 实现斜向斑马线边框 要实现斜向斑马线间隔圆环,我们可以使用css paint api。该api提供了强大的功能,可以让我们在元素上绘制复杂的图形。 立即学习“前端…

    2025年12月24日
    000
  • 图片如何不撑高父容器?

    如何让图片不撑高父容器? 当父容器包含不同高度的子元素时,父容器的高度通常会被最高元素撑开。如果你希望父容器的高度由文本内容撑开,避免图片对其产生影响,可以通过以下 css 解决方法: 绝对定位元素: .child-image { position: absolute; top: 0; left: …

    2025年12月24日
    000
  • 为什么自定义样式表在 Safari 中访问百度页面时无法生效?

    自定义样式表在 safari 中失效的原因 用户尝试在 safari 偏好设置中添加自定义样式表,代码如下: body { background-image: url(“/users/luxury/desktop/wallhaven-o5762l.png”) !important;} 测试后发现,在…

    2025年12月24日
    000
  • CSS 帮助

    我正在尝试将文本附加到棕色框的左侧。我不能。我不知道代码有什么问题。请帮助我。 css .hero { position: relative; bottom: 80px; display: flex; justify-content: left; align-items: start; color:…

    2025年12月24日 好文分享
    200
  • 前端代码辅助工具:如何选择最可靠的AI工具?

    前端代码辅助工具:可靠性探讨 对于前端工程师来说,在HTML、CSS和JavaScript开发中借助AI工具是司空见惯的事情。然而,并非所有工具都能提供同等的可靠性。 个性化需求 关于哪个AI工具最可靠,这个问题没有一刀切的答案。每个人的使用习惯和项目需求各不相同。以下是一些影响选择的重要因素: 立…

    2025年12月24日
    300
  • 如何用 CSS Paint API 实现倾斜的斑马线间隔圆环?

    实现斑马线边框样式:探究 css paint api 本文将探究如何使用 css paint api 实现倾斜的斑马线间隔圆环。 问题: 给定一个有多个圆圈组成的斑马线图案,如何使用 css 实现倾斜的斑马线间隔圆环? 答案: 立即学习“前端免费学习笔记(深入)”; 使用 css paint api…

    2025年12月24日
    000

发表回复

登录后才能评论
关注微信