How to kill CompletableFuture related threads?

后端 未结 2 1899
-上瘾入骨i
-上瘾入骨i 2021-01-07 06:16

I have method that is checking the CompletableFuture execution time. If such CompletableFuture is executing for more than 2 seconds i want to kill this task. But how can I d

相关标签:
2条回答
  • 2021-01-07 06:50

    When you create a chain of CompletableFuture stages like b = a.thenApply(function), this handy method creates a setup of different components. Basically, these components refer to each other as a → function → b, so the completion of a will trigger the evaluation of function which will first pre-check whether b still is not completed, then evaluate your function and attempt to complete b with the result.

    But b itself has no knowledge of function or the thread that will evaluate it. In fact, function is not special to b, anyone could call complete, completeExceptionally or cancel on it from any thread, the first one winning. Hence, the completable in the class name.

    The only way to get hands on the threads evaluating the functions, is to be in control of them right from the start, e.g.

    ExecutorService myWorkers = Executors.newFixedThreadPool(2);
    
    CompletableFuture<FinalResultType> future
        = CompletableFuture.supplyAsync(() -> generateInitialValue(), myWorkers)
                           .thenApplyAsync(v -> nextCalculation(v), myWorkers)
                           .thenApplyAsync(v -> lastCalculation(v), myWorkers);
    future.whenComplete((x,y) -> myWorkers.shutdownNow());
    

    Now, the completion of future, e.g. via cancellation, will ensure that no new evaluation will be triggered by this chain and further makes an attempt to interrupt ongoing evaluations, if any.

    So you can implement a timeout, e.g.

    try {
        try {
            FinalResultType result = future.get(2, TimeUnit.SECONDS);
            System.out.println("got "+result);
        }
        catch(TimeoutException ex) {
            if(future.cancel(true)) System.out.println("cancelled");
            else System.out.println("got "+future.get());
        }
    }
    catch(ExecutionException|InterruptedException ex) {
        ex.printStackTrace();
    }
    

    Not that the rejection of tasks due to the shutdown of the thread pool may cause some of the intermediate future to never complete, but for this chain of stages, this is irrelevant. All that matters, is, that the final stage future is completed, which is guaranteed, as it is its completion which triggers the shutdown.

    0 讨论(0)
  • 2021-01-07 07:13

    The only way to terminate a thread is via interruption, which is a cooperative mechanism. This means the the thread must implement interruption logic, by handling the InterruptedException.

    But it is a really bad practice to interrupt threads that you don't own, which I think is your case.

    0 讨论(0)
提交回复
热议问题