6.4 ThreadPoolExecutor线程池的关闭方法shutdown和shutdownNow区别、示例

shutdown():

把线程池的状态设置成SHUTDOWN状态,然后中断所有没有正执行任务的线程

shutdownNow():

首先把线程池的状态设置成STOP,然后尝试停止所有正在执行任务或者暂停任务的线程,并返回等待执行任务的列表

常用

通常我们调用shutdown()方法关闭线程池,如果不需要任务执行完成,可以调用shutdownNow()

原理

遍历线程池中的工作线程,然后组个调用线程的interrupt方法来中断线程,所以无法响应中断任务的可能永远无法停止

其他方法

isShutdown():

只要调用了shutdown或者shutdownNow,isShutdown就会返回true,否则false

isTerminaed():

当所有任务关闭之后,才表示线程池关闭成功,这时会返回true

isTerminating:

执行了shutdown或shutdownNow之后,还有任务正在进行中=true,没有任务进行中=false

测试示例

测试shutdown

public class Test {
    static class Customer extends Thread {
        @Override
        public void run() {
            System.out.println(Thread.currentThread().getName() + "开始执行");
            try {
                //模拟任务执行4s
                Thread.sleep(4000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName() + "执行完成");
        }
    }

    public static void main(String[] args) throws ExecutionException, InterruptedException {

        ThreadPoolExecutor executor = new ThreadPoolExecutor(
                10,
                20,
                20,
                TimeUnit.SECONDS,
                new ArrayBlockingQueue<Runnable>(10), new ThreadFactory() {
            @Override
            public Thread newThread(Runnable r) {
                //1、设置线程的名字
                Thread thread = new Thread(r);
                //thread.setName("通过自定义线程工厂谁知线程名字");
                //2、设置成守护线程
                //thread.setDaemon(true);
                return thread;
            }
        },
                //四种阻塞队列满之后的处理方式
//                new ThreadPoolExecutor.CallerRunsPolicy()); 使用调用线程处理,比如是main调用的线程池,用main线程执行
//                new ThreadPoolExecutor.DiscardOldestPolicy());把阻塞队列首元素丢掉,执行当前线程
//                new ThreadPoolExecutor.DiscardPolicy());直接丢掉当前线程
                new AbortPolicy());//跑出异常

        executor.execute(new Customer());
        Thread.sleep(100);
        //当前线程还未执行完,执行shutdown
        System.out.println("执行shutdown之前");
        System.out.println(executor.isTerminated());
        System.out.println(executor.isShutdown());
        System.out.println(executor.isTerminating());

        executor.shutdown();
        System.out.println("执行shutdown之后,但是还有任务正在执行");
        //线程还没有执行完成,isTerminated()=false
        System.out.println(executor.isTerminated());
        System.out.println(executor.isShutdown());
        System.out.println("有任务进行中isTerminating="+executor.isTerminating());

        //线程执行完成,isTerminated()=true
        Thread.sleep(4000);
        System.out.println("执行shutdown之后,所有任务执行完成");
        System.out.println(executor.isTerminated());
        System.out.println("没有任务进行中isTerminating="+executor.isTerminating());
    }
}

结果

Thread-1开始执行
执行shutdown之前
false
false
false
执行shutdown之后,但是还有任务正在执行
false
true
有任务进行中isTerminating=true
Thread-1执行完成
执行shutdown之后,所有任务执行完成
true
没有任务进行中isTerminating=false

测试shutdownNow

public class Test {
    static class Customer extends Thread {
        @Override
        public void run() {
            System.out.println(Thread.currentThread().getName() + "开始执行");
            try {
                //模拟任务执行4s
                Thread.sleep(4000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName() + "执行完成");
        }
    }

    public static void main(String[] args) throws ExecutionException, InterruptedException {

        ThreadPoolExecutor executor = new ThreadPoolExecutor(
                10,
                20,
                20,
                TimeUnit.SECONDS,
                new ArrayBlockingQueue<Runnable>(10), new ThreadFactory() {
            @Override
            public Thread newThread(Runnable r) {
                //1、设置线程的名字
                Thread thread = new Thread(r);
                //thread.setName("通过自定义线程工厂谁知线程名字");
                //2、设置成守护线程
                //thread.setDaemon(true);
                return thread;
            }
        },
                //四种阻塞队列满之后的处理方式
//                new ThreadPoolExecutor.CallerRunsPolicy()); 使用调用线程处理,比如是main调用的线程池,用main线程执行
//                new ThreadPoolExecutor.DiscardOldestPolicy());把阻塞队列首元素丢掉,执行当前线程
//                new ThreadPoolExecutor.DiscardPolicy());直接丢掉当前线程
                new AbortPolicy());//跑出异常

        executor.execute(new Customer());
        Thread.sleep(100);
        //当前线程还未执行完,执行shutdown
        System.out.println("执行shutdown之前");
        System.out.println(executor.isTerminated());
        System.out.println(executor.isShutdown());
        System.out.println(executor.isTerminating());

        executor.shutdownNow();
        System.out.println("执行shutdown之后,但是还有任务正在执行");
        //线程还没有执行完成,isTerminated()=false
        System.out.println(executor.isTerminated());
        System.out.println(executor.isShutdown());
        System.out.println("有任务进行中isTerminating="+executor.isTerminating());

        //线程执行完成,isTerminated()=true
        Thread.sleep(4000);
        System.out.println("执行shutdown之后,所有任务执行完成");
        System.out.println(executor.isTerminated());
        System.out.println("没有任务进行中isTerminating="+executor.isTerminating());
    }
}

结果:因为要中断正在执行的任务,所有会报中断异常

Thread-1开始执行
执行shutdown之前
false
false
false
执行shutdown之后,但是还有任务正在执行
false
true
有任务进行中isTerminating=true
java.lang.InterruptedException: sleep interrupted
	at java.lang.Thread.sleep(Native Method)
	at cn.enjoyedu.ch6.Test$Customer.run(Test.java:14)
	at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142)
	at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617)
	at java.lang.Thread.run(Thread.java:748)
Thread-1执行完成
执行shutdown之后,所有任务执行完成
true
没有任务进行中isTerminating=false

  • 3
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
ThreadPoolExecutorJava中的一个线程池实现类。它继承自ExecutorService接口,可以用来管理和执行线程任务。ThreadPoolExecutor线程池提供了更灵活的线程管理和任务调度的功能,并且可以根据需要进行配置。可以通过指定核心线程数、最大线程数、线程存活时间和任务队列等参数来创建和配置ThreadPoolExecutor线程池。 使用ThreadPoolExecutor线程池可以提供以下几个优点: 1. 降低线程创建和销毁的开销。线程池可以重用已经创建的线程,减少了频繁创建和销毁线程的开销。 2. 提高系统的响应速度。线程池可以并发执行多个任务,提高了系统的处理能力和响应速度。 3. 控制线程并发数量。通过设置线程池的核心线程数和最大线程数,可以控制系统的并发线程数量,避免资源耗尽和系统崩溃的风险。 4. 提供任务调度和管理。线程池可以将任务按照一定的策略和优先级进行调度和执行,方便管理任务的执行顺序和优先级。 总之,ThreadPoolExecutor线程池是一个灵活可配置的线程管理和任务调度工具,可以提高系统的并发处理能力和响应速度。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *3* [线程池ThreadPoolExecutor详解(整理详细)](https://blog.csdn.net/trusause/article/details/125747447)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *2* [ThreadPoolExecutor线程池的使用方法](https://download.csdn.net/download/weixin_38659648/12746355)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值