我在CompletableFuture的supplyAsync()中处理长时间运行的操作,并将结果放入thenAccept()中。有时thenAccept()在主线程上执行,但有时它在工作线程上运行。但是我只想在主线程上运行thenAccept()操作。这是示例代码。

private void test() {

    ExecutorService executorService = Executors.newSingleThreadExecutor();

    CompletableFuture<String> cf1 = CompletableFuture.supplyAsync(() -> {
        System.out.println("supplyAsync | I am running on : " + Thread.currentThread().getName());
        return "Hello world";
    }, executorService);

    CompletableFuture<Void> cf3 = cf1.thenAccept(s -> {
        System.out.print("thenAccept | I am running on : " + Thread.currentThread().getName());
        System.out.println(" | answer : " + s);
    });

    cf3.thenRun(() -> {
        System.out.println("thenRun | I am running on : " + Thread.currentThread().getName());
        System.out.println();
    });

}

public static void main(String[] args) {

    App app = new App();
    for(int i = 0; i < 3; i++){
        app.test();
    }
}

结果是:
supplyAsync | I am running on : pool-1-thread-1
thenAccept | I am running on : main | answer : Hello world
thenRun | I am running on : main

supplyAsync | I am running on : pool-2-thread-1
thenAccept | I am running on : main | answer : Hello world
thenRun | I am running on : main

supplyAsync | I am running on : pool-3-thread-1
thenAccept | I am running on : pool-3-thread-1 | answer : Hello world
thenRun | I am running on : pool-3-thread-1

我怎样才能解决这个问题 ?

最佳答案

看看CompletableFuture的JavaDoc。有趣的部分是有关 CompletionStage 策略的部分。

在这里,您发现使用非异步方法会导致一种或非情形。然后,如果您看一下实现,您将最终遇到Java Runtime的非公开部分。存在一些 UNSAFE 处理,这意味着可能发生某种竞赛情况。

我建议使用 thenAcceptAsync() thenRunAsync()变体,并将executorService变量传递给这两个调用。

09-28 09:51