Java程序  |  148行  |  5.13 KB

/*
 * Copyright (C) 2014 The Guava Authors
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.google.common.util.concurrent;

import static com.google.common.base.Preconditions.checkNotNull;

import com.google.caliper.Benchmark;
import com.google.caliper.Param;
import com.google.caliper.api.VmOptions;
import com.google.common.collect.ImmutableList;

import java.util.concurrent.Callable;
import java.util.concurrent.CancellationException;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Executor;

/**
 * A benchmark for {@link Futures#combine}
 */
@VmOptions({"-Xms12g", "-Xmx12g", "-d64"})
public class FuturesCombineBenchmark {

  enum Impl {
    OLD {
      @Override <V> ListenableFuture<V> combine(final Callable<V> combiner, Executor executor,
          Iterable<? extends ListenableFuture<?>> futures) {
        ListenableFuture<?> trigger = Futures.successfulAsList(futures);
        checkNotNull(combiner);
        checkNotNull(trigger);
        return Futures.transform(trigger, new AsyncFunction<Object, V>() {
          @Override public ListenableFuture<V> apply(Object arg) throws Exception {
            try {
              return Futures.immediateFuture(combiner.call());
            } catch (CancellationException e) {
              return Futures.immediateCancelledFuture();
            } catch (ExecutionException e) {
              return Futures.immediateFailedFuture(e.getCause()); // OK to rethrow on Error
            }
          }
        }, executor);
      }
    },
    NEW {
      @Override
      <V> ListenableFuture<V> combine(Callable<V> combiner, final Executor executor,
          Iterable<? extends ListenableFuture<?>> futures) {
        return Futures.combine(combiner, executor, futures);
      }
    };

    abstract <V> ListenableFuture<V> combine(
        Callable<V> combiner, Executor executor,
        Iterable<? extends ListenableFuture<?>> futures);
  }

  private static final Executor INLINE_EXECUTOR = new Executor() {
    @Override public void execute(Runnable command) {
      command.run();
    }
  };

  @Param Impl impl;
  @Param({"1", "5", "10"}) int numInputs;

  @Benchmark int timeDoneSuccesfulFutures(int reps) throws Exception {
    ImmutableList.Builder<ListenableFuture<?>> futuresBuilder = ImmutableList.builder();
    for (int i = 0; i < numInputs; i++) {
      futuresBuilder.add(Futures.immediateFuture(i));
    }
    ImmutableList<ListenableFuture<?>> futures = futuresBuilder.build();
    Impl impl = this.impl;
    Callable<Integer> callable = Callables.returning(12);
    int sum = 0;
    for (int i = 0; i < reps; i++) {
      sum += impl.combine(callable, INLINE_EXECUTOR, futures).get();
    }
    return sum;
  }
  
  @Benchmark int timeDoneFailedFutures(int reps) throws Exception {
    ImmutableList.Builder<ListenableFuture<?>> futuresBuilder = ImmutableList.builder();
    for (int i = 0; i < numInputs; i++) {
      futuresBuilder.add(Futures.immediateFailedFuture(new Exception("boom")));
    }
    ImmutableList<ListenableFuture<?>> futures = futuresBuilder.build();
    Impl impl = this.impl;
    Callable<Integer> callable = Callables.returning(12);
    int sum = 0;
    for (int i = 0; i < reps; i++) {
      sum += impl.combine(callable, INLINE_EXECUTOR, futures).get();
    }
    return sum;
  }
  
  @Benchmark int timeSuccesfulFutures(int reps) throws Exception {
    Impl impl = this.impl;
    Callable<Integer> callable = Callables.returning(12);
    int sum = 0;
    for (int i = 0; i < reps; i++) {
      ImmutableList<SettableFuture<Integer>> futures = getSettableFutureList();
      ListenableFuture<Integer> combined = impl.combine(callable, INLINE_EXECUTOR, futures);
      for (SettableFuture<Integer> future : futures) {
        future.set(i);
      }
      sum += combined.get();
    }
    return sum;
  }

  @Benchmark int timeFailedFutures(int reps) throws Exception {
    Impl impl = this.impl;
    Callable<Integer> callable = Callables.returning(12);
    int sum = 0;
    Exception throwable = new Exception("boom");
    for (int i = 0; i < reps; i++) {
      ImmutableList<SettableFuture<Integer>> futures = getSettableFutureList();
      ListenableFuture<Integer> combined = impl.combine(callable, INLINE_EXECUTOR, futures);
      for (SettableFuture<Integer> future : futures) {
        future.setException(throwable);
      }
      sum += combined.get();
    }
    return sum;
  }

  private ImmutableList<SettableFuture<Integer>> getSettableFutureList() {
    ImmutableList.Builder<SettableFuture<Integer>> futuresBuilder = ImmutableList.builder();
    for (int i = 0; i < numInputs; i++) {
      futuresBuilder.add(SettableFuture.<Integer>create());
    }
    return futuresBuilder.build();
  }
}