org.dbflute.helper.thread.CountDownRace Maven / Gradle / Ivy
/*
* Copyright 2014-2015 the original author or 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 org.dbflute.helper.thread;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.Callable;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import org.dbflute.helper.thread.exception.ThreadFireFailureException;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* @author jflute
* @since 1.0.5A (2013/10/17 Thursday)
*/
public class CountDownRace {
// ===================================================================================
// Definition
// ==========
private static final Logger _log = LoggerFactory.getLogger(CountDownRace.class);
// ===================================================================================
// Attribute
// =========
protected final int _runnerCount;
// ===================================================================================
// Constructor
// ===========
public CountDownRace(int runnerCount) {
if (runnerCount < 1) {
String msg = "The argument 'runnerCount' should not be minus or zero: " + runnerCount;
throw new IllegalArgumentException(msg);
}
_runnerCount = runnerCount;
}
// ===================================================================================
// Thread Fire
// ===========
public void readyGo(CountDownRaceExecution oneArgLambda) {
if (oneArgLambda == null) {
String msg = "The argument 'execution' should be not null.";
throw new IllegalArgumentException(msg);
}
doReadyGo(oneArgLambda);
}
protected void doReadyGo(CountDownRaceExecution execution) {
final ExecutorService service = Executors.newCachedThreadPool();
final CountDownLatch ready = new CountDownLatch(_runnerCount);
final CountDownLatch start = new CountDownLatch(1);
final CountDownLatch goal = new CountDownLatch(_runnerCount);
final CountDownRaceLatch ourLatch = new CountDownRaceLatch(_runnerCount);
final Object lockObj = new Object();
final List> futureList = new ArrayList>();
for (int i = 0; i < _runnerCount; i++) { // basically synchronized with parameter size
final int entryNumber = i + 1;
final Callable callable = createCallable(execution, ready, start, goal, ourLatch, entryNumber, lockObj);
final Future future = service.submit(callable);
futureList.add(future);
}
if (_log.isDebugEnabled()) {
_log.debug("...Ready Go! Count Down Race just begun! (runner=" + _runnerCount + ")");
}
start.countDown(); // fire!
try {
goal.await(); // wait until all threads are finished
if (_log.isDebugEnabled()) {
_log.debug("All runners finished line! (runner=" + _runnerCount + ")");
}
} catch (InterruptedException e) {
String msg = "goal.await() was interrupted!";
throw new IllegalStateException(msg, e);
}
handleFuture(futureList);
}
protected void handleFuture(final List> futureList) {
for (Future future : futureList) {
try {
future.get();
} catch (InterruptedException e) {
String msg = "future.get() was interrupted!";
throw new IllegalStateException(msg, e);
} catch (ExecutionException e) {
String msg = "Failed to fire the thread: " + future;
throw new ThreadFireFailureException(msg, e.getCause());
}
}
}
// ===================================================================================
// Callable
// ========
protected Callable createCallable(final CountDownRaceExecution execution, final CountDownLatch ready, final CountDownLatch start,
final CountDownLatch goal, final CountDownRaceLatch ourLatch, final int entryNumber, final Object lockObj) {
return new Callable() {
public Void call() { // each thread here
final long threadId = Thread.currentThread().getId();
try {
ready.countDown();
try {
start.await();
} catch (InterruptedException e) {
String msg = "start.await() was interrupted: start=" + start;
throw new IllegalStateException(msg, e);
}
RuntimeException cause = null;
try {
execution.execute(createRunner(threadId, ourLatch, entryNumber, lockObj));
} catch (RuntimeException e) {
cause = e;
}
if (cause != null) {
throw cause;
}
} finally {
goal.countDown();
ourLatch.reset(); // to release waiting threads
}
return null;
}
};
}
protected CountDownRaceRunner createRunner(long threadId, CountDownRaceLatch ourLatch, int entryNumber, Object lockObj) {
return new CountDownRaceRunner(threadId, ourLatch, entryNumber, lockObj, _runnerCount);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy