This is the 18th day of my participation in Gwen Challenge
The background,
Recently I have been learning about concurrent programming thread pool. I have read many articles and videos, but I think none of them talk about the core of thread pool. Everyone says thread pooling reduces thread creation and destruction time, but doesn’t explain how it works, just keeps repeating what happens if it’s bigger than the core thread count, what happens if it’s bigger than the maximum thread count. It’s like putting the cart before the horse. I thought I had to look into it, otherwise thread pools would seem redundant to me.
Two, the key code
First of all, let’s clarify the order of execution behind the execute method. I will only list them in order, and only specify the key points when I know them:
-
execute()
-
addworker(w)
-
Work a = new Worker(w)
Work overrides the run method, which is runwork()
-
a.start()
The above is the logical order in which a thread is added to the thread pool, which is not reusable. Work overrides run and runs inside runwork () : this is the key.
final void runWorker(Worker w) {
Thread wt = Thread.currentThread();
Runnable task = w.firstTask;
w.firstTask = null;
w.unlock(); // allow interrupts
boolean completedAbruptly = true;
try {
// Task is not empty the first time it comes in
// Gettask is executed when task is null due to final execution
while(task ! =null|| (task = getTask()) ! =null) {
w.lock();
if((runStateAtLeast(ctl.get(), STOP) || (Thread.interrupted() && runStateAtLeast(ctl.get(), STOP))) && ! wt.isInterrupted()) wt.interrupt();try {
beforeExecute(wt, task);
Throwable thrown = null;
try {
task.run();
} catch (RuntimeException x) {
thrown = x; throw x;
} catch (Error x) {
thrown = x; throw x;
} catch (Throwable x) {
thrown = x; throw new Error(x);
} finally{ afterExecute(task, thrown); }}finally {
// empty!!
task = null;
w.completedTasks++;
w.unlock();
}
}
completedAbruptly = false;
} finally{ processWorkerExit(w, completedAbruptly); }}Copy the code
What’s the essence of this: You can see that the thread’s run method is a while loop where getTask blocks, which we’ll talk about later, so the thread doesn’t end.
Here’s what getTask was doing:
private Runnable getTask(a) {
boolean timedOut = false; // Did the last poll() time out?
for (;;) {
int c = ctl.get();
int rs = runStateOf(c);
// Check if queue empty only if necessary.
if (rs >= SHUTDOWN && (rs >= STOP || workQueue.isEmpty())) {
decrementWorkerCount();
return null;
}
int wc = workerCountOf(c);
// Are workers subject to culling?
boolean timed = allowCoreThreadTimeOut || wc > corePoolSize;
if ((wc > maximumPoolSize || (timed && timedOut))
&& (wc > 1 || workQueue.isEmpty())) {
if (compareAndDecrementWorkerCount(c))
return null;
continue;
}
try {
// Do not worry about the previous line, look at the judgment in this line.
Timed True if the number of current threads is greater than the number of core threads
// If the number of current threads is greater than the number of core threads, the current thread is queued
// If the current number of threads is less than the number of core threads, go to the queue
Runnable r = timed ?
workQueue.poll(keepAliveTime, TimeUnit.NANOSECONDS) :
workQueue.take();
if(r ! =null)
return r;
timedOut = true;
} catch (InterruptedException retry) {
timedOut = false; }}}Copy the code
Let’s say that thread A, when it first comes in, will add it to the Work array, execute run, and get to gettask and find that the queue is empty. His queue.take() is therefore blocked (so the thread will not end), another thread B, executes again and then blocks at queue.tak(), another thread C, since it is now larger than the core thread count, passes the offer method of a queue (see the code below), puts the thread into the queue, So the take of the first two threads can get the thread, so reuse! The c thread does not go through start (), so the thread creation time is reduced! Wonderful!
public void execute(Runnable command) {
if (command == null)
throw new NullPointerException();
int c = ctl.get();
if (workerCountOf(c) < corePoolSize) {
if (addWorker(command, true))
return;
c = ctl.get();
}
// When the third thread enters, the number of core threads is greater than the number of core threads, so the offer!
// Obviously, but the offer fails, i.e., the queue is full, and the next else if is sent
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);
}
// As long as it does not exceed the Max thread, it will continue to create threads, a new thread, directly run start
else if(! addWorker(command,false))
reject(command);
}
Copy the code
Third, summary
Think back to this: the essence of reuse is that the while loop keeps the run method going and then blocks with the queue.