前言
最近在复习的时候,发现一个运行线程池拒绝策略demo中的main方法在运行了之后,进程并没有关闭。看了jconsole线程池中的线程都处于waiting状态。这里是跟我设置线程池的线程工厂中的设置线程是否为后台线程有关。
后台线程和非后台线程
后台线程,也叫守护线程,指的是在程序运行的时候后台提供一种通用服务的线程,比如jvm里垃圾回收线程,这种线程并不属于程序中不可或缺的部分。因此,当所有的非守护线程结束时,程序也就终止了,同时会杀死进程中的所有守护线程。反过来说,只要有任何非守护线程在运行,程序就不会终止。
守护线程和非守护线程的区别:在于jvm的离开:如果用户线程(非守护线程)已经全部退出运行了,只剩下守护线程存在,那么虚拟机也就退出了。因为没了被守护者,守护线程也就没有工作可做了。
设置守护线程是通过调用Thread对象的setDaemon(true)方法来实现的。在使用守护线程的时候需要注意以下几点:
(1)thread.setDeaemon(true)必须在thread.start()之前设置,否则会报错IllegalThreadStateException
(2)在Daemon线程中产生的线程也是Daemon的。
(3)守护线程应该永远不去访问固有资源,如文件、数据库,因为它会在任何时候甚至在一个操作的中间发生终端。
不退出程序的代码
这里在main方法结束之后不能正常退出的代码是测试了线程池两种抛出拒绝策略的场景:
(1)线程池处于SHUTDOWN状态时,再提交新的任务到线程池
(2)线程池中所有的线程都处于运行状态,并且阻塞队列已满,这时再去提交新的线程。
public class ThreadPoolExecutorRejectNewTaskDemo {
/**
* 线程池的最大容量
*/
private static final int MAX_POOL_SIZE = 3;
/**
* 阻塞队列的容量
*/
private static final int QUEUE_CAPACITY = 2;
/**
* 非核心线程处于空闲状态的最长时间
*/
private static final int KEEP_ALIVE_TIME = 1;
/**
* 线程池对象 注意这里传入的是默认的拒绝策略 也就是AbortPolicy
*/
private static final ThreadPoolExecutor threadPool = new ThreadPoolExecutor(MAX_POOL_SIZE, MAX_POOL_SIZE, KEEP_ALIVE_TIME,
TimeUnit.SECONDS, new LinkedBlockingDeque<>(QUEUE_CAPACITY), new MyThreadFactory("task-reject"));
public static void main(String[] args) {
// shutdownThreadPoolToRejectNewTask();
fullQueueThreadPoolRejectNewTask();
}
/**
* 模拟线程池被关闭之后, 继续向其中提交新任务被拒绝的场景
*/
private static void shutdownThreadPoolToRejectNewTask() {
MyRunnable runnable = new MyRunnable()