package org.infinispan.executors; import java.util.Collection; import java.util.Collections; import java.util.List; import java.util.concurrent.Callable; import java.util.concurrent.ExecutionException; import java.util.concurrent.Future; import java.util.concurrent.ScheduledExecutorService; import java.util.concurrent.ScheduledFuture; import java.util.concurrent.ThreadFactory; import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeoutException; import org.infinispan.commons.executors.ThreadPoolExecutorFactory; /** * A delegating scheduled executor that lazily constructs and initializes the underlying scheduled executor, since * unused JDK executors are expensive. * * @author Manik Surtani * @since 5.1 */ public class LazyInitializingScheduledExecutorService implements ScheduledExecutorService { private volatile ScheduledExecutorService delegate; private final ThreadPoolExecutorFactory<ScheduledExecutorService> executorFactory; private final ThreadFactory threadFactory; public LazyInitializingScheduledExecutorService( ThreadPoolExecutorFactory<ScheduledExecutorService> executorFactory, ThreadFactory threadFactory) { this.executorFactory = executorFactory; this.threadFactory = threadFactory; } private void initIfNeeded() { if (delegate == null) { synchronized (this) { if (delegate == null) { delegate = executorFactory.createExecutor(threadFactory); } } } } @Override public void shutdown() { if (delegate != null) delegate.shutdown(); } @Override public List<Runnable> shutdownNow() { if (delegate == null) return Collections.emptyList(); else return delegate.shutdownNow(); } @Override public boolean isShutdown() { return delegate == null || delegate.isShutdown(); } @Override public boolean isTerminated() { return delegate == null || delegate.isTerminated(); } @Override public boolean awaitTermination(long timeout, TimeUnit unit) throws InterruptedException { if (delegate == null) return true; else return delegate.awaitTermination(timeout, unit); } @Override public <T> Future<T> submit(Callable<T> task) { initIfNeeded(); return delegate.submit(task); } @Override public <T> Future<T> submit(Runnable task, T result) { initIfNeeded(); return delegate.submit(task, result); } @Override public Future<?> submit(Runnable task) { initIfNeeded(); return delegate.submit(task); } @Override public <T> List<Future<T>> invokeAll(Collection<? extends Callable<T>> tasks) throws InterruptedException { initIfNeeded(); return delegate.invokeAll(tasks); } @Override public <T> List<Future<T>> invokeAll(Collection<? extends Callable<T>> tasks, long timeout, TimeUnit unit) throws InterruptedException { initIfNeeded(); return delegate.invokeAll(tasks, timeout, unit); } @Override public <T> T invokeAny(Collection<? extends Callable<T>> tasks) throws InterruptedException, ExecutionException { initIfNeeded(); return delegate.invokeAny(tasks); } @Override public <T> T invokeAny(Collection<? extends Callable<T>> tasks, long timeout, TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException { initIfNeeded(); return delegate.invokeAny(tasks, timeout, unit); } @Override public void execute(Runnable command) { initIfNeeded(); delegate.execute(command); } @Override public ScheduledFuture<?> schedule(Runnable command, long delay, TimeUnit unit) { initIfNeeded(); return delegate.schedule(command, delay, unit); } @Override public <V> ScheduledFuture<V> schedule(Callable<V> callable, long delay, TimeUnit unit) { initIfNeeded(); return delegate.schedule(callable, delay, unit); } @Override public ScheduledFuture<?> scheduleAtFixedRate(Runnable command, long initialDelay, long period, TimeUnit unit) { initIfNeeded(); return delegate.scheduleAtFixedRate(command, initialDelay, period, unit); } @Override public ScheduledFuture<?> scheduleWithFixedDelay(Runnable command, long initialDelay, long delay, TimeUnit unit) { initIfNeeded(); return delegate.scheduleWithFixedDelay(command, initialDelay, delay, unit); } }