在写上一篇文章的时候就在想,executor.shutDown会不会把正在执行中的任务给中断了?虽然当时也分析过,但是当时确实只是泛泛而读,在实际中还是相当的模糊。借此机会,我们再复习一下。
代码语言:javascript复制 public void shutdown() {
final ReentrantLock mainLock = this.mainLock;
mainLock.lock();
try {
checkShutdownAccess();
//将线程池状态设置为shutdown
advanceRunState(SHUTDOWN);
//中断线程池中的worker线程
interruptIdleWorkers();
onShutdown(); // hook for ScheduledThreadPoolExecutor
} finally {
mainLock.unlock();
}
tryTerminate();
}
那么中断线程的操作就在这个interruptIdleWorkers()中。
代码语言:javascript复制 private void interruptIdleWorkers(boolean onlyOne) {
final ReentrantLock mainLock = this.mainLock;
mainLock.lock();
try {
for (Worker w : workers) {
Thread t = w.thread;
//这里的tryLock是尝试获取工作线程的锁,如果工作线程在运行
//那么就获取不到锁,也就是无法中断。所以这里中断的都是没有运行的线程
if (!t.isInterrupted() && w.tryLock()) {
try {
//调用每个线程的中断
t.interrupt();
} catch (SecurityException ignore) {
} finally {
w.unlock();
}
}
if (onlyOne)
break;
}
} finally {
mainLock.unlock();
}
}
所以说shutdown是不会让线程直接退出的,之前添加的线程还是会运行到自然结束。但是因为这块设置了线程池状态为shutdown,因此线程是添加不了的。
而在shutdownNow方法中
代码语言:javascript复制 public List<Runnable> shutdownNow() {
List<Runnable> tasks;
final ReentrantLock mainLock = this.mainLock;
mainLock.lock();
try {
checkShutdownAccess();
//直接设置为stop状态
advanceRunState(STOP);
//中断线程
interruptWorkers();
tasks = drainQueue();
} finally {
mainLock.unlock();
}
tryTerminate();
return tasks;
}
代码语言:javascript复制 private void interruptWorkers() {
final ReentrantLock mainLock = this.mainLock;
mainLock.lock();
try {
for (Worker w : workers)
//逐个进行中断
w.interruptIfStarted();
} finally {
mainLock.unlock();
}
}
代码语言:javascript复制 void interruptIfStarted() {
Thread t;
if (getState() >= 0 && (t = thread) != null && !t.isInterrupted()) {
try {
//直接调用中断
t.interrupt();
} catch (SecurityException ignore) {
}
}
}
所以看的出来,ShutDownNow是直接中断所有的工作线程。并且阻止新线程的添加。