Q1:JVM退出时会不会关闭线程池?
不会。JVM退出时会kill所有的进程内的线程。
怎么优雅关闭线程池?
shutdown方法将线程池状态从running->shutdown,这个状态下会拒绝所有提交的任务,直接调用拒绝策略。
会尝试中断所有空闲线程。让剩余线程执行完当前任务并执行完队列中剩余任务。
那shutdown什么时候推进状态呢?
是在最后一个工作线程退出时会执行processWorkerExit,调用tryTerminate,先将状态推进到tyding,表示线程池正在清理中。
其实这个tyding状态对线程池本身没什么逻辑执行,只是提供了terminated接口给用户去自定义逻辑做些清理事情。
回调完terminated方法后再将tyding推进到terminated。此时所有工作线程消亡,队列也没有任务,线程池算是终结了。
shutdownNow方法将状态从running->stop,此时中断所有线程无论是否空闲还是工作中,抽出队列所有任务待返回给用户。
直接执行tryTerminate方法。
JVM提供了钩子函数,在退出时回调这些钩子。在钩子中可以对线程池执行关闭。
会开一个线程执行回调,并且等到钩子函数执行完毕JVM才会kill所有线程然后退出。
对钩子函数要谨慎编程,否则造成JVM无法退出的情况。
Runtime.getRuntime().addShutdownHook(new Thread() { // do something });
参考文章:
JVM关闭的时候到底会不会等待线程池线程任务执行完毕_wen-pan的博客-CSDN博客
JVM关闭的时候到底会不会等待线程池线程任务执行完毕