CompletableFuture、CompletionStage 和 Future 的对比

1. 共同点

  • 异步编程: 这三种机制都用于处理异步操作,即在不阻塞当前线程的情况下执行任务,并在任务完成后通知结果。

  • 结果获取: 所有三种机制都提供获取异步操作结果的方法,例如:

    • Future.get()
    • CompletionStage.toCompletableFuture().join()
    • CompletableFuture.join()
  • 错误处理: 所有三种机制都提供处理异步操作中发生的异常的方法,例如:

    • Future.get() 可能会抛出异常。
    • CompletionStage.exceptionally()CompletableFuture.exceptionally() 可以捕获异常。

2. 优势

Future

  • 简单易用: Future 是 Java 1.5 中引入的,相对来说比较简单,易于理解。
  • 基础机制: Future 是异步操作的核心,其他的机制,例如 CompletionStageCompletableFuture 都是基于 Future 实现的。

CompletionStage

  • 灵活的组合和处理: CompletionStage 提供了丰富的组合方法,例如 thenApplythenAcceptthenCompose 等,可以方便地将多个异步操作进行组合和处理,以满足复杂业务需求。
  • 链式调用: CompletionStage 支持链式调用,可以将多个异步操作连接起来,形成一个完整的异步操作流程,简化代码编写。

CompletableFuture

  • 功能更丰富: CompletableFuture 提供了更加丰富的方法,除了 CompletionStage 接口中定义的方法之外,还提供了 completecompleteExceptionallyjoin 等方法,可以更加灵活地控制异步操作。
  • 高效: CompletableFuture 的实现更加高效,因为它使用了一些优化技术,例如延迟初始化和缓存。

3. 示例代码

3.1、Future

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;

public class FutureExample {

    public static void main(String[] args) throws Exception {
        ExecutorService executor = Executors.newSingleThreadExecutor();
        Future<String> future = executor.submit(() -> {
            // 模拟耗时操作
            Thread.sleep(1000);
            return "Hello, Future!";
        });

        // 获取结果
        String result = future.get();
        System.out.println("Result: " + result);

        executor.shutdown();
    }
}

3.2、CompletionStage

import java.util.concurrent.CompletableFuture;

public class CompletionStageExample {

    public static void main(String[] args) {
        CompletionStage<String> future = CompletableFuture.supplyAsync(() -> {
            // 模拟耗时操作
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            return "Hello, CompletionStage!";
        });

        future.thenApply(s -> s.toUpperCase())
                .thenAccept(System.out::println)
                .exceptionally(throwable -> {
                    System.err.println("Error occurred: " + throwable.getMessage());
                    return null;
                });
    }
}

进阶用法

3.3、、将两个 CompletionStage 的结果合并在一起
我们以一个常见的场景 — 网络请求为例,来说明 CompletionStage 的链式调用如何简化代码:
场景描述:
假设你需要从两个不同的 API 获取数据,然后将这两个数据进行合并处理,最后将合并后的数据展示给用户。
传统方法:

// 使用传统的回调方式
public void fetchAndProcessData() {
    // 第一个 API 请求
    api1.fetch(new Callback<Data1>() {
        @Override
        public void onSuccess(Data1 data1) {
            // 第二个 API 请求
            api2.fetch(new Callback<Data2>() {
                @Override
                public void onSuccess(Data2 data2) {
                    // 合并数据
                    ProcessedData processedData = processData(data1, data2);
                    // 展示结果
                    showResult(processedData);
                }

                @Override
                public void onFailure(Throwable throwable) {
                    // 处理错误
                    handleError(throwable);
                }
            });
        }

        @Override
        public void onFailure(Throwable throwable) {
            // 处理错误
            handleError(throwable);
        }
    });
}

使用 CompletionStage 的方法:

// 使用 CompletionStage 进行链式调用
public void fetchAndProcessData() {
    CompletionStage<Data1> data1Future = api1.fetch();
    CompletionStage<Data2> data2Future = api2.fetch();

    data1Future.thenCombine(data2Future, this::processData) // 合并数据
        .thenAccept(this::showResult) // 展示结果
        .exceptionally(this::handleError); // 处理错误
}

private ProcessedData processData(Data1 data1, Data2 data2) {
    // 合并数据逻辑
    // ... 
    return processedData;
}

private void showResult(ProcessedData processedData) {
    // 展示结果逻辑
    // ...
}

private Throwable handleError(Throwable throwable) {
    // 处理错误逻辑
    // ...
    return throwable;
}

3.4、将 CompletionStage 的结果作为另一个 CompletionStage 的输入

示例:
假设我们有一个数据库查询操作,需要先获取用户 ID,然后再根据用户 ID 获取用户信息。我们可以使用 CompletionStage 的 thenCompose 方法来实现这个操作:

import java.util.concurrent.CompletableFuture;
import java.util.function.Function;

public class CompletionStageChain {

    public static void main(String[] args) {
        // Step 1: 获取用户信息
        CompletableFuture<String> user = CompletableFuture.supplyAsync(() -> {
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            return "用户A";
        });

        // Step 2: 使用用户信息获取订单列表
        Function<String, CompletableFuture<String>> getOrders = userStr -> CompletableFuture.supplyAsync(() -> {
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            return "订单列表:" + userStr + "的订单信息";
        });

        // Step 3: 将用户信息和订单列表拼接在一起
        Function<String, String> joinUserAndOrders = orders -> "用户信息:" + user.join() + "\n" + orders;

        // 将多个 CompletionStage 连接起来
        user.thenCompose(getOrders) 
                .thenApply(joinUserAndOrders)
                .thenAccept(System.out::println)
                .exceptionally(throwable -> {
                    System.err.println("Error occurred: " + throwable.getMessage());
                    return null;
                });
    }
}

解释:

  • 在上面的示例中,我们使用 thenApplyuser 的结果作为 getOrders 函数的输入,从而获得 订单列表
  • 接着,我们使用 thenApply订单列表 作为 joinUserAndOrders 函数的输入,从而将用户信息和订单列表拼接在一起。
  • 最后,我们使用 thenAccept 将最终结果输出到控制台。

简化代码

  • 使用CompletionStage的链式调用,我们可以将多个异步操作连接起来,形成一个完整的流程,代码更加简洁易读。
  • 相比于手动管理多个 Future 对象,CompletionStage 的链式调用可以更方便地处理异步操作的组合和处理,提高代码的可读性和可维护性。

4.执行效率

细心者会发现,上面经常用到CompletableFuture.supplyAsync 方法来开始一个异步任务,也恰恰就是在supplyAsync中,会使用一个异步线程来执行传入的 Supplier 函数。但是,这个异步线程并非由 CompletableFuture 自己创建,而是来自于 Java 中的 ForkJoinPool。

image.png
image.png

ForkJoinPool

定义: ForkJoinPool 是 Java 提供的一个线程池,专门设计用于执行可分解为更小任务的并行任务。

工作原理: ForkJoinPool 使用了一种分治的思想,将一个任务分解为多个子任务,并使用多个线程并行执行这些子任务,最终将结果合并,实现高效的并行计算。

使用场景: ForkJoinPool 适用于处理那些可以分解为更小任务的任务,例如:

  • 并行计算
  • 海量数据处理
  • 递归算法

CompletableFuture.supplyAsync 和 ForkJoinPool

CompletableFuture.supplyAsync 方法默认使用 ForkJoinPool.commonPool() 来执行异步任务。
ForkJoinPool.commonPool() 是一个共享的 ForkJoinPool,它由所有使用 ForkJoinPool 的类共享使用。
如果你需要更精细的控制,你也可以通过 CompletableFuture.supplyAsync(supplier, executor) 方法指定一个自定义的线程池来执行异步任务。

5. 总结

  • Future 是一个基础的异步操作接口,简单易用。
  • CompletionStage 提供了更加灵活的异步操作组合和处理方法,支持链式调用,简化代码编写。
  • CompletableFutureCompletionStage 的基础上提供了更丰富的功能,更方便的控制和操作,并且更加高效。

根据实际需求选择合适的异步操作机制,可以有效提高程序的效率和可维护性。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,013评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,205评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,370评论 0 342
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,168评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,153评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,954评论 1 283
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,271评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,916评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,382评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,877评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,989评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,624评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,209评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,199评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,418评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,401评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,700评论 2 345

推荐阅读更多精彩内容