publicvoidexecute(Runnable command){ if (command == null) thrownew 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) { //取出当前执行的线程数量,如果小于线程池大小,加入新的worker if (addWorker(command, true)) return; //加入成功,直接返回 c = ctl.get(); //加入失败,取出最新的数据,因为在这期间worker的数量可能已经发生变化了 } if (isRunning(c) && workQueue.offer(command)) { //如果当前线程池处于RUNNING状态,并且任务队列可以成功加入任务 int recheck = ctl.get(); //二次确认 if (! isRunning(recheck) && remove(command)) //如果当前线程池不是RUNNING状态,则从队列中移除任务 reject(command); //拒绝任务 elseif (workerCountOf(recheck) == 0) //该情况考虑的是线程池是SHUTDOWN状态,但是队列还有任务没有完成,需要加一个null任务进入线程池,避免任务进入 addWorker(null, false); } //如果添加任务失败,拒绝任务 elseif (!addWorker(command, false)) reject(command); }
privatebooleanaddWorker(Runnable firstTask, boolean core){ retry: for (;;) { int c = ctl.get(); int rs = runStateOf(c); //获取当前线程池的状态
// Check if queue empty only if necessary. if (rs >= SHUTDOWN && ! (rs == SHUTDOWN && firstTask == null && ! workQueue.isEmpty())) returnfalse; //如果状态不是RUNNING,或者状态是SHUTDOWN,队列第一个任务是空,并且队列不为空,直接返回false
for (;;) { int wc = workerCountOf(c); //获取到workerCount的大小 if (wc >= CAPACITY || //如果超过了CAPACITY或者超过设定的线程池大小 wc >= (core ? corePoolSize : maximumPoolSize)) returnfalse; if (compareAndIncrementWorkerCount(c)) //尝试对c增加1,里面使用了自旋锁,加1因为加在了低位, break retry; //跳出循环 c = ctl.get(); // Re-read ctl if (runStateOf(c) != rs) //如果当前状态与上面取的不一样,说明发生了变化,重新开始循环 continue retry; // else CAS failed due to workerCount change; retry inner loop } }
boolean workerStarted = false; boolean workerAdded = false; Worker w = null; //初始化worker的信息 try { w = new Worker(firstTask); final Thread t = w.thread; //新建一个worker if (t != null) { final ReentrantLock mainLock = this.mainLock; //获取锁,同时只能有一个线程对ctl进行修改 mainLock.lock(); try { // Recheck while holding lock. // Back out on ThreadFactory failure or if // shut down before lock acquired. int rs = runStateOf(ctl.get()); //再次校验线程池的状态
if (rs < SHUTDOWN || (rs == SHUTDOWN && firstTask == null)) { //如果线程池处于RUNNING状态,或者处于SHUTDOWN但是传入的任务是null if (t.isAlive()) // 如果新加入的线程已经开始工作,并且还没有结束,抛出异常 thrownew IllegalThreadStateException(); workers.add(w); //将当前的worker加入到worker列表 int s = workers.size(); //更改最大的线程池大小 if (s > largestPoolSize) largestPoolSize = s; workerAdded = true; } } finally { mainLock.unlock(); //释放锁 } if (workerAdded) { t.start(); workerStarted = true; //修改标志位 } } } finally { if (! workerStarted) addWorkerFailed(w); } return workerStarted; //返回线程工作状态 }
privatefinalclassWorker extendsAbstractQueuedSynchronizer implementsRunnable { /** * This class will never be serialized, but we provide a * serialVersionUID to suppress a javac warning. */ privatestaticfinallong serialVersionUID = 6138294804551838833L;
/** Thread this worker is running in. Null if factory fails. */ final Thread thread; /** Initial task to run. Possibly null. */ Runnable firstTask; /** Per-thread task counter */ volatilelong completedTasks;
/** * Creates with given first task and thread from ThreadFactory. * @param firstTask the first task (null if none) */ Worker(Runnable firstTask) { setState(-1); // inhibit interrupts until runWorker this.firstTask = firstTask; this.thread = getThreadFactory().newThread(this); }
/** Delegates main run loop to outer runWorker */ publicvoidrun(){ runWorker(this); }
// Lock methods // // The value 0 represents the unlocked state. // The value 1 represents the locked state.