/** * AnalyzerBeans * Copyright (C) 2014 Neopost - Customer Information Management * * This copyrighted material is made available to anyone wishing to use, modify, * copy, or redistribute it subject to the terms and conditions of the GNU * Lesser General Public License, as published by the Free Software Foundation. * * This program 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 distribution; if not, write to: * Free Software Foundation, Inc. * 51 Franklin Street, Fifth Floor * Boston, MA 02110-1301 USA */ package org.eobjects.analyzer.job.concurrent; import java.util.concurrent.ArrayBlockingQueue; import java.util.concurrent.BlockingQueue; import java.util.concurrent.ExecutorService; import java.util.concurrent.RejectedExecutionException; import java.util.concurrent.ThreadFactory; import java.util.concurrent.ThreadPoolExecutor; import java.util.concurrent.TimeUnit; import org.eobjects.analyzer.job.tasks.Task; import org.slf4j.Logger; import org.slf4j.LoggerFactory; /** * The preferred {@link TaskRunner} implementation based on the * java.util.concurrent package (specifically the {@link ExecutorService} * class). */ public final class MultiThreadedTaskRunner implements TaskRunner { private static final Logger logger = LoggerFactory.getLogger(MultiThreadedTaskRunner.class); private final ThreadFactory _threadFactory; private final ExecutorService _executorService; private final int _numThreads; private final BlockingQueue<Runnable> _workQueue; public MultiThreadedTaskRunner() { this(30); } public MultiThreadedTaskRunner(int numThreads) { _numThreads = numThreads; // if all threads are busy, newly submitted tasks will by run by caller final ThreadPoolExecutor.CallerRunsPolicy rejectionHandler = new ThreadPoolExecutor.CallerRunsPolicy(); // there will be a minimum task capacity of 20, and preferably // numThreads * 3 (to avoid blocking buffer behaviour) final int taskCapacity = Math.max(20, numThreads * 3); _threadFactory = new DaemonThreadFactory(); _workQueue = new ArrayBlockingQueue<Runnable>(taskCapacity); _executorService = new ThreadPoolExecutor(numThreads, numThreads, 60, TimeUnit.SECONDS, _workQueue, _threadFactory, rejectionHandler); } /** * @return the amount of threads in the thread pool, or -1 if this * information is not available */ public int getNumThreads() { return _numThreads; } @Override public void run(final Task task, final TaskListener listener) { logger.debug("run({},{})", task, listener); executeInternal(new TaskRunnable(task, listener)); } @Override public void run(TaskRunnable taskRunnable) { logger.debug("run({})", taskRunnable); executeInternal(taskRunnable); } private void executeInternal(TaskRunnable taskRunnable) { try { _executorService.execute(taskRunnable); } catch (RejectedExecutionException e) { logger.error("Unexpected rejected execution!", e); } } @Override public void shutdown() { logger.info("shutdown() called, shutting down executor service"); _executorService.shutdown(); } public ExecutorService getExecutorService() { return _executorService; } @Override protected void finalize() throws Throwable { shutdown(); } @Override public void assistExecution() { Runnable task = _workQueue.poll(); if (task != null) { task.run(); } } }