前言
最近在复习的时候,发现一个运行线程池拒绝策略demo中的main方法在运行了之后,进程并没有关闭。看了jconsole线程池中的线程都处于waiting状态。这里是跟我设置线程池的线程工厂中的设置线程是否为后台线程有关。
后台线程和非后台线程
后台线程,也叫守护线程,指的是在程序运行的时候后台提供一种通用服务的线程,比如jvm里垃圾回收线程,这种线程并不属于程序中不可或缺的部分。因此,当所有的非守护线程结束时,程序也就终止了,同时会杀死进程中的所有守护线程。反过来说,只要有任何非守护线程在运行,程序就不会终止。
守护线程和非守护线程的区别:在于jvm的离开:如果用户线程(非守护线程)已经全部退出运行了,只剩下守护线程存在,那么虚拟机也就退出了。因为没了被守护者,守护线程也就没有工作可做了。
设置守护线程是通过调用Thread对象的setDaemon(true)方法来实现的。在使用守护线程的时候需要注意以下几点:
(1)thread.setDeaemon(true)必须在thread.start()之前设置,否则会报错IllegalThreadStateException
(2)在Daemon线程中产生的线程也是Daemon的。
(3)守护线程应该永远不去访问固有资源,如文件、数据库,因为它会在任何时候甚至在一个操作的中间发生终端。
不退出程序的代码
这里在main方法结束之后不能正常退出的代码是测试了线程池两种抛出拒绝策略的场景:
(1)线程池处于SHUTDOWN状态时,再提交新的任务到线程池
(2)线程池中所有的线程都处于运行状态,并且阻塞队列已满,这时再去提交新的线程。
1 | public class ThreadPoolExecutorRejectNewTaskDemo { |
这段代码中创建新线程的线程工厂中设置了线程池中的线程是否为非后台线程,这里设置的是非后台线程,在我们运行main方法之后,即可观察到main方法一直没退出,而线程池中的线程都处于waiting状态。(这里需要设置下不达到拒绝策略的条件,比如去掉达到拒绝策略的条件之后不再提交任务),这里只是复现这个程序不退出的场景。这里只要把新创建的线程设置为后台线程,当main方法结束之后,线程池中的线程都是守护线程,会随着用户线程的结束也被结束掉。
同样的,我们可以看下Executors提供的一些工具线程池,比如Executors.newFixedThreadPool(int nThreads)
它的构造是使用的默认的线程工厂,我们可以看看默认线程工厂中线程的设置是否为后台线程:
1 | /** |
可以看到默认的线程工厂在创建新的线程的时候也是设置daemon为false,这使得定长线程池中都是用户线程,这些个在主线程结束之后,也会使程序不能退出。