To understand this, you will need to see what is happening when you are submitting the task to the Executor in a loop.
First we will just look at the submission of a single task to the Executor. I will now be referring to the JDK 1.7.0_51
source code
The static method Executor.newFixedThreadPool
method returns a ThreadPoolExecutor
containing a blocking queue to hold the task
public static ExecutorService newFixedThreadPool(int nThreads) {
return new ThreadPoolExecutor(nThreads, nThreads,
0L, TimeUnit.MILLISECONDS,
new LinkedBlockingQueue<Runnable>());
}
The moment you add a task to this Executor,this goes to the submit method of ThreadPoolExecutor
extending AbstractExecutorService
where the implementation of submit method is written.
public <T> Future<T> submit(Callable<T> task) {
if (task == null) throw new NullPointerException();
RunnableFuture<T> ftask = newTaskFor(task);
execute(ftask);
return ftask;
}
The execute method is implementation specific ( meaning different types of Executor implement it different ways)
Now comes the real meat. This is the execute method defined in ThreadPoolExecutor
. Especially pay attention at the comments.
Here few configuration parameters of ThreadPoolExecutor like corePoolSize
comes into play.
public void execute(Runnable command) {
if (command == null)
throw new NullPointerException();
/*
* Proceed in 3 steps:
*
* 1. If fewer than corePoolSize threads are running, try to
* start a new thread with the given command as its first
* task. The call to addWorker atomically checks runState and
* workerCount, and so prevents false alarms that would add
* threads when it shouldn't, by returning false.
*
* 2. If a task can be successfully queued, then we still need
* to double-check whether we should have added a thread
* (because existing ones died since last checking) or that
* the pool shut down since entry into this method. So we
* recheck state and if necessary roll back the enqueuing if
* stopped, or start a new thread if there are none.
*
* 3. If we cannot queue task, then we try to add a new
* thread. If it fails, we know we are shut down or saturated
* and so reject the task.
*/
int c = ctl.get();
if (workerCountOf(c) < corePoolSize) {
if (addWorker(command, true))
return;
c = ctl.get();
}
if (isRunning(c) && workQueue.offer(command)) {
int recheck = ctl.get();
if (! isRunning(recheck) && remove(command))
reject(command);
else if (workerCountOf(recheck) == 0)
addWorker(null, false);
}
else if (!addWorker(command, false))
reject(command);
}