Skip to content

Commit

Permalink
ISPN-1817 Lazily initialize executors and scheduled executors
Browse files Browse the repository at this point in the history
  • Loading branch information
maniksurtani authored and Sanne committed Feb 8, 2012
1 parent 8c27dd6 commit 88c7323
Show file tree
Hide file tree
Showing 3 changed files with 308 additions and 2 deletions.
@@ -0,0 +1,139 @@
/*
* Copyright 2012 Red Hat, Inc. and/or its affiliates.
*
* This is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as
* published by the Free Software Foundation; either version 2.1 of
* the License, or (at your option) any later version.
*
* This software is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
* 02110-1301 USA
*/

package org.infinispan.executors;

import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.Properties;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;

/**
* A delegating executor that lazily constructs and initalizes the underlying executor, since unused JDK executors
* are expensive.
*
* @author Manik Surtani
* @since 5.1
*/
public class LazyInitializingExecutorService implements ExecutorService {
private ExecutorService delegate;
private final ExecutorFactory factory;
private final Properties executorProperties;

public LazyInitializingExecutorService(ExecutorFactory factory, Properties executorProperties) {
this.factory = factory;
this.executorProperties = executorProperties;
}

private void initIfNeeded() {
if (delegate == null) {
synchronized (this) {
if (delegate == null) {
delegate = factory.getExecutor(executorProperties);
}
}
}
}

@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);
}
}
@@ -0,0 +1,165 @@
/*
* Copyright 2012 Red Hat, Inc. and/or its affiliates.
*
* This is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as
* published by the Free Software Foundation; either version 2.1 of
* the License, or (at your option) any later version.
*
* This software is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
* 02110-1301 USA
*/

package org.infinispan.executors;

import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.Properties;
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.TimeUnit;
import java.util.concurrent.TimeoutException;

/**
* A delegating scheduled executor that lazily constructs and initalizes the underlying scheduled executor, since
* unused JDK executors are expensive.
*
* @author Manik Surtani
* @since 5.1
*/
public class LazyInitializingScheduledExecutorService implements ScheduledExecutorService {

private ScheduledExecutorService delegate;
private final ScheduledExecutorFactory factory;
private final Properties executorProperties;

public LazyInitializingScheduledExecutorService(ScheduledExecutorFactory factory, Properties executorProperties) {
this.factory = factory;
this.executorProperties = executorProperties;
}

private void initIfNeeded() {
if (delegate == null) {
synchronized (this) {
if (delegate == null) {
delegate = factory.getScheduledExecutor(executorProperties);
}
}
}
}

@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);
}
}
Expand Up @@ -24,6 +24,8 @@

import org.infinispan.config.ConfigurationException;
import org.infinispan.executors.ExecutorFactory;
import org.infinispan.executors.LazyInitializingExecutorService;
import org.infinispan.executors.LazyInitializingScheduledExecutorService;
import org.infinispan.executors.ScheduledExecutorFactory;
import org.infinispan.factories.annotations.DefaultFactoryFor;
import org.infinispan.factories.annotations.Stop;
Expand Down Expand Up @@ -116,7 +118,7 @@ private ExecutorService buildAndConfigureExecutorService(String factoryName, Pro
setComponentName(componentName, props);
setDefaultThreads(KnownComponentNames.getDefaultThreads(componentName), props);
setDefaultThreadPrio(KnownComponentNames.getDefaultThreadPrio(componentName), props);
return f.getExecutor(props);
return new LazyInitializingExecutorService(f, props);
}

private ScheduledExecutorService buildAndConfigureScheduledExecutorService(String factoryName, Properties p, String componentName) throws Exception {
Expand All @@ -125,7 +127,7 @@ private ScheduledExecutorService buildAndConfigureScheduledExecutorService(Strin
ScheduledExecutorFactory f = (ScheduledExecutorFactory) Util.getInstance(factoryName, globalConfiguration.getClassLoader());
setComponentName(componentName, props);
setDefaultThreadPrio(KnownComponentNames.getDefaultThreadPrio(componentName), props);
return f.getScheduledExecutor(props);
return new LazyInitializingScheduledExecutorService(f, props);
}

private void setDefaultThreadPrio(int prio, Properties props) {
Expand Down

0 comments on commit 88c7323

Please sign in to comment.