ListenableFuture,FutureCallback和超时

问题描述 投票:33回答:1

根据番石榴的例子,我看到我一直在寻找解决问题的优雅方案。具体来说,我喜欢Futures.addCallback(ListenableFuture, FutureCallback) 的工作方式,但是我希望能够在调用FutureCallback之前的时间长度上设置超时。最佳如果违反超时只会导致调用FutureCallback的失败条件会很好。

番石榴有没有这样的东西?是不是不建议尝试将超时与回调相结合?

编辑:包括引导我到这一点的代码示例。显然,我删除了有意义的位以获得最小的例子。

@Test
public void testFuture()
{
    Callable<Boolean> callable = new Callable<Boolean>()
    {

        @Override
        public Boolean call() throws Exception
        {
            while(true);
        }
    };

    ListenableFuture<Boolean> callableFuture = MoreExecutors.listeningDecorator(Executors.newSingleThreadExecutor()).submit(callable);

    Futures.addCallback(callableFuture, new FutureCallback<Boolean>()
    {

        @Override
        public void onFailure(Throwable arg0)
        {
            System.out.println("onFailure:"+arg0); 
        }

        @Override
        public void onSuccess(Boolean arg0)
        {
            System.out.println("onSuccess:"+arg0);
        }
    }); 

    try
    {
        callableFuture.get(1000, TimeUnit.MILLISECONDS);
    }catch(Throwable t)
    {
        System.out.println("catch:"+t);
    }
}

此代码只会打印catch:java.util.concurrent.TimeoutException

java guava
1个回答
23
投票

更新:这已被添加到番石榴作为Futures.withTimeout()


在内部,我们有一个makeTimeoutFuture方法,它将Future作为输入并返回一个新的Future,除非原始未在给定的截止日期之前完成,否则将具有相同的结果。如果截止日期到期,则输出Future将其结果设置为TimeoutException。所以,你可以调用makeTimeoutFuture并将监听器附加到输出Future

makeTimeoutFuture不是解决您问题的最自然的解决方案。实际上,我认为该方法的创建主要是为了在no-arg get()调用上设置硬超时,因为将所需的截止时间传播给所有调用者可能会很麻烦。一个更自然的解决方案是推测get()get(long, TimeUnit),因为addCallback(ListenableFuture, FutureCallback)addCallback(ListenableFuture, FutureCallback, long, TimeUnit, SchededuledExecutorService)。这有点笨拙,尽管不如makeTimeoutFuture。在做出任何承诺之前,我想更多地考虑这个问题。你会file a feature request吗?

(这是我们内部的:)

public static <V> ListenableFuture<V> makeTimeoutFuture(
    ListenableFuture<V> delegate,
    Duration duration,
    ScheduledExecutorService scheduledExecutor)

如果指定的持续时间到期,则返回委托给另一个但将提前完成的未来(通过TimeoutException中包含的ExecutionException)。在这种情况下,代表未来不会被取消。

scheduledExecutor.schedule(new Runnable() {
  @Override public void run() {
    TimeoutFuture.this.setException(new TimeoutException("Future timed out"));
  }
}, duration.getMillis(), TimeUnit.MILLISECONDS);
© www.soinside.com 2019 - 2024. All rights reserved.