Jdk21优雅处理异步任务

摘要:本文主要介绍jdk21 如何优雅的处理异步执行任务;下面给出了几个常见的案例。

CompletableFuture解决回调地狱写法

顺序执行(前阶段结果传递给后阶段)

适用于需要按顺序执行异步操作的场景(如:先查用户,再查订单)。

方法 描述
thenApply(Function) 同步处理前阶段结果,返回新结果(有入参、有返回值)。
thenAccept(Consumer) 同步消费前阶段结果,无返回值(有入参、无返回值)。
thenRun(Runnable) 前阶段完成后执行操作,不依赖结果(无入参、无返回值)。
thenCompose(Function) 合并两个阶段为链式调用(前阶段返回CompletableFuture,直接衔接后阶段)。

案例

csharp 复制代码
    public void test01(){
        CompletableFuture<OrderDto> orderDtoCompletableFuture = CompletableFuture.supplyAsync(() -> {
            OrderDto orderDto = new OrderDto();
            System.out.println(Thread.currentThread().getName());
            orderDto.setId(1L);
            return orderDto;
        }).thenApplyAsync(orderDto -> {
            try{
                Thread.sleep(2000L);
            }catch (Exception ex){

            }
            orderDto.setName("测试");
            System.out.println("模拟执行耗时,查询商品信息");
            return orderDto;
        }).thenApplyAsync(orderDto -> {
            try{
                Thread.sleep(2000L);
            }catch (Exception ex){

            }
            orderDto.setMoney(new BigDecimal("20"));
            System.out.println("模拟执行耗时,计算商品总价");
            return orderDto;
        }).exceptionally(ex -> {
            log.error("执行异常", ex);
            return null;
        });
        OrderDto or = orderDtoCompletableFuture.join();
        System.out.println(or);
        System.out.println("主线程执行完了");
    }

并行执行(多阶段独立执行后合并)

适用于并行处理多个任务后合并结果的场景(如:同时计算用户积分和统计订单)。

方法 描述
thenCombine(CompletableFuture, BiFunction) 等待两个阶段都完成,合并结果(前阶段结果 + 后阶段结果 → 新结果)。
allOf(CompletableFuture...) 等待所有阶段完成(无返回值,需手动收集结果)。
anyOf(CompletableFuture...) 等待任意一个阶段完成(返回第一个完成的结果)。

案例

ini 复制代码
// 阶段 1:计算用户积分(异步)
CompletableFuture<Integer> pointsFuture = CompletableFuture.supplyAsync(() -> 
    calculatePoints(user)
);

// 阶段 2:统计有效订单数(异步)
CompletableFuture<Integer> validOrdersFuture = CompletableFuture.supplyAsync(() -> 
    countValidOrders(user)
);

// 合并两个阶段的结果(总积分 = 积分 + 订单数×10)
CompletableFuture<Integer> totalFuture = pointsFuture.thenCombine(
    validOrdersFuture, 
    (points, orders) -> points + orders * 10
);

int total = totalFuture.join(); // 输出总结果

异常处理

链式调用中若某阶段抛出异常,后续阶段会被静默取消,需通过以下方法捕获异常:

方法 描述
exceptionally(Function) 异常时返回默认值(类似 try-catch 的 catch 块)。
handle(BiFunction) 无论成功或失败都处理(可返回新结果或默认值)。

案例

arduino 复制代码
CompletableFuture<String> future = CompletableFuture.supplyAsync(() -> {
    if (condition) throw new RuntimeException("查询失败");
    return "正常数据";
}).exceptionally(ex -> {
    System.out.println("异常处理:" + ex.getMessage());
    return "默认数据"; // 异常时返回默认值
});

String result = future.join(); // 若异常,返回 "默认数据"

虚线程使用案例

虚线程异步执行并等待结果

ini 复制代码
public class VirtualThreadCompletableFutureDemo {
    public static void main(String[] args) {
        ExecutorService virtualExecutor = Executors.newVirtualThreadPerTaskExecutor();
        List<CompletableFuture<Void>> futures = new ArrayList<>();

        // 提交 3 个虚线程任务
        for (int i = 0; i < 3; i++) {
            int taskId = i;
            CompletableFuture<Void> future = CompletableFuture.runAsync(() -> {
                try {
                    Thread.sleep(1000); // 模拟任务执行
                    System.out.println("任务 " + taskId + " 完成");
                } catch (InterruptedException e) {
                    Thread.currentThread().interrupt();
                }
            }, virtualExecutor);
            futures.add(future);
        }

        // 判断所有任务是否完成(阻塞等待)
        CompletableFuture.allOf(futures.toArray(new CompletableFuture[0])).join();

        System.out.println("所有任务完成!");
        virtualExecutor.shutdown();
    }
}

结果

复制代码
任务 0 完成
任务 1 完成
任务 2 完成
所有任务完成!

扩展:单个任务异常判断

arduino 复制代码
CompletableFuture<String> future = CompletableFuture.supplyAsync(() -> {
    throw new RuntimeException("任务失败"); // 模拟异常
}, virtualExecutor);

future.whenComplete((result, ex) -> {
    if (ex != null) {
        System.out.println("任务异常:" + ex.getMessage()); // 输出:任务异常:任务失败
    }
});

// 主动检查是否异常完成
if (future.isCompletedExceptionally()) {
    System.out.println("任务因异常终止");
}
相关推荐
一只乔哇噻1 分钟前
java后端工程师进修ing(研一版 || day41)
java·开发语言·学习·算法
愿时间能学会宽恕2 分钟前
SpringBoot后端开发常用工具详细介绍——SpringSecurity认证用户保证安全
spring boot·后端·安全
CodeSheep17 分钟前
稚晖君又开始摇人了,有点猛啊!
前端·后端·程序员
User_芊芊君子19 分钟前
【Java】设计模式——单例、工厂、代理模式
java·设计模式·代理模式
小宁爱Python21 分钟前
Django 从环境搭建到第一个项目
后端·python·django
2301_8035545228 分钟前
正向代理,反向代理,负载均衡还有nginx
java·nginx·负载均衡
要开心吖ZSH29 分钟前
软件设计师备考-(十六)数据结构及算法应用(重要)
java·数据结构·算法·软考·软件设计师
uzong30 分钟前
深入浅出:画好技术图
后端·架构
向上的车轮37 分钟前
基于Java Spring Boot的云原生TodoList Demo 项目,验证云原生核心特性
java·spring boot·云原生
IT_陈寒39 分钟前
Java性能优化:从这8个关键指标开始,让你的应用提速50%
前端·人工智能·后端