ThreadPoolExecutor 线程执行超时,释放线程

简介: ThreadPoolExecutor 线程执行超时,释放线程

如果线程中的执行时间过长,导致长时间被占用,可以通过新建一个子线程,来监控主线程的执行超时时间,如果超时了,通过子线程杀掉父线程 (主意,父线程被杀后,子线程还会活着) 子线程杀掉主线程

这个问题其实还是没有搞定。下面的代码只是发起了线程的中断,某一行代码执行结束后,不会执行后续的代码。但就这某一行卡住了的话,本方案还是无解的。

package com.vipsoft.Thread;
import java.util.concurrent.*;
import java.util.concurrent.atomic.AtomicInteger;
public class ThreadPoolExecutorTest {
    public static void main(String[] args) throws Exception {
        int corePoolSize = 2;
        int maximumPoolSize = 5;
        long keepAliveTime = 10;
        TimeUnit unit = TimeUnit.SECONDS;
        BlockingQueue<Runnable> workQueue = new LinkedBlockingQueue<>(3);  //定义一个大小为2的队列,只等有一个任务在排队等,多出来的需要开新线程
        ThreadFactory threadFactory = new MyTreadFactory();
        RejectedExecutionHandler handler = new MyPolicy();
        ThreadPoolExecutor executor = new ThreadPoolExecutor(corePoolSize, maximumPoolSize, keepAliveTime, unit, workQueue, threadFactory, handler);
        System.out.println("预启动线程(备战)");
        executor.prestartAllCoreThreads(); // 预启动所有核心线程,处于备战
        System.out.println("预启动线程数(备战):" + executor.getPoolSize());
        for (int i = 1; i <= 10; i++) {
            System.out.println(System.currentTimeMillis() + "  " + "开始 下发任务:" + i + " 当前线程总数:" + executor.getPoolSize());
            MyTask task = new MyTask(String.valueOf(i));
            executor.execute(task);
            System.out.println(System.currentTimeMillis() + "  " + "完成 下发任务:" + i + " 当前线程总数:" + executor.getPoolSize() + " 队列中的线程数量:" + workQueue.size());
            Thread.sleep(1); //停1毫秒,日志记录,时间后方便分析
            if (i == 9) {
                //TODO Thread.sleep(3000); //任务9下发后【会被拒绝】,停3秒,等队列或线程释放后,再下发任务10,这时候任务10不会被拒绝
            }
        }
        System.in.read(); //阻塞主线程
    }
    static class MyTreadFactory implements ThreadFactory {
        private final AtomicInteger mThreadNum = new AtomicInteger(1);
        @Override
        public Thread newThread(Runnable r) {
            Thread t = new Thread(r, "Thread-" + mThreadNum.getAndIncrement());
            System.out.println(System.currentTimeMillis() + "  " + t.getName() + " has been created");
            return t;
        }
    }
    public static class MyPolicy implements RejectedExecutionHandler {
        public void rejectedExecution(Runnable r, ThreadPoolExecutor e) {
            // 可做日志记录等
            System.err.println(System.currentTimeMillis() + "  " + r.toString() + " rejected from " + e.toString());
        }
    }
    static class MyTask implements Runnable {
        private String name;
        public MyTask(String name) {
            this.name = name;
        }
        @Override
        public void run() {
            try {
                Thread mainT = Thread.currentThread();
                String taskName = this.name;
                SubThread st = new SubThread(taskName, mainT);
                Thread thread = new Thread(st);
                thread.start();
                try {
                    long millis = 1000;
                    if (Integer.valueOf(taskName) % 2 == 0) {
                        millis = Integer.valueOf(taskName) * 2 * millis; //让任务执行慢点
                    }
                    System.out.println(System.currentTimeMillis() + "  " + this.toString() + " 开始运行! 运行线程 " + Thread.currentThread().getName() + " 需要(秒):" + millis);
                    Thread.sleep(millis); //让任务执行慢点--这边还是有点问题,如果这个 sleep 过长的话,还是需要等它执行完,并不能解决线程超时杀掉它的目的
                    System.out.println(System.currentTimeMillis() + "  " + this.toString() + " 运行结束! 运行线程 " + Thread.currentThread().getName());
                } catch (InterruptedException e) {
                    System.out.println(System.currentTimeMillis() + "  " + this.toString() + " 被打断 运行线程 " + Thread.currentThread().getName());
                } finally {
                    st.setDone(); //通知监控线程,主线程运行结束
                    //TODO thread.interrupt(); //父线程也可以中止子线程的任务执行
                }
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
        @Override
        public String toString() {
            return "MyTask [name=" + this.name + "]";
        }
        static class SubThread implements Runnable {
            private String name;
            private Thread mainT;
            //如果完成了就不用杀了。
            private boolean isDone;
            public void setDone() {
                isDone = true;
            }
            public SubThread(String name, Thread mainT) {
                this.name = name;
                this.mainT = mainT;
            }
            @Override
            public void run() {
                try {
                    System.out.println(System.currentTimeMillis() + " 子线程 " + Thread.currentThread().getName() + "【正在】监控主进程 " + mainT.getName() + " 任务" + name);
                    Thread.sleep(5000);
                    //时间到了,主线程还没有完成就干掉。
                    if (!isDone) {
                        mainT.interrupt();
                        System.out.println(System.currentTimeMillis() + " 子线程 " + Thread.currentThread().getName() + "【杀掉】主进程(超时了) " + mainT.getName() + " 任务" + name);
                    }
                } catch (InterruptedException e) {
                    System.out.println(System.currentTimeMillis() + " 子线程 " + Thread.currentThread().getName() + "【被打断】" + mainT.getName() + " 任务" + name);
                } finally {
                    System.out.println(System.currentTimeMillis() + " 子线程 " + Thread.currentThread().getName() + "【完成】监控主进程 " + mainT.getName() + " 任务" + name);
                }
            }
            @Override
            public String toString() {
                return "MyTask [name=" + this.name + "]";
            }
        }
    }
}

 

日志排序后:

1655083181083  Thread-1 has been created
1655083181083  Thread-2 has been created
1655083181083  开始 下发任务:1 当前线程总数:2
1655083181084  完成 下发任务:1 当前线程总数:2 队列中的线程数量:1
1655083181085  MyTask [name=1] 开始运行! 运行线程 Thread-2 需要(秒):1000
1655083181085 子线程 Thread-0【正在】监控主进程 Thread-2 任务1
1655083181086  MyTask [name=2] 开始运行! 运行线程 Thread-1 需要(秒):4000
1655083181086  开始 下发任务:2 当前线程总数:2
1655083181086  完成 下发任务:2 当前线程总数:2 队列中的线程数量:1
1655083181086 子线程 Thread-1【正在】监控主进程 Thread-1 任务2
1655083181088  开始 下发任务:3 当前线程总数:2
1655083181088  完成 下发任务:3 当前线程总数:2 队列中的线程数量:1
1655083181089  开始 下发任务:4 当前线程总数:2
1655083181089  完成 下发任务:4 当前线程总数:2 队列中的线程数量:2
1655083181091  开始 下发任务:5 当前线程总数:2
1655083181091  完成 下发任务:5 当前线程总数:2 队列中的线程数量:3
1655083181093  MyTask [name=6] 开始运行! 运行线程 Thread-3 需要(秒):12000
1655083181093  Thread-3 has been created
1655083181093  开始 下发任务:6 当前线程总数:2
1655083181093  完成 下发任务:6 当前线程总数:3 队列中的线程数量:3
1655083181093 子线程 Thread-2【正在】监控主进程 Thread-3 任务6
1655083181095  MyTask [name=7] 开始运行! 运行线程 Thread-4 需要(秒):1000
1655083181095  Thread-4 has been created
1655083181095  开始 下发任务:7 当前线程总数:3
1655083181095  完成 下发任务:7 当前线程总数:4 队列中的线程数量:3
1655083181095 子线程 Thread-3【正在】监控主进程 Thread-4 任务7
1655083181097  MyTask [name=8] 开始运行! 运行线程 Thread-5 需要(秒):16000
1655083181097  Thread-5 has been created
1655083181097  开始 下发任务:8 当前线程总数:4
1655083181097  完成 下发任务:8 当前线程总数:5 队列中的线程数量:3
1655083181097 子线程 Thread-4【正在】监控主进程 Thread-5 任务8
1655083181099  MyTask [name=9] rejected from java.util.concurrent.ThreadPoolExecutor@4b1210ee[Running, pool size = 5, active threads = 5, queued tasks = 3, completed tasks = 0]
1655083181099  开始 下发任务:9 当前线程总数:5
1655083181099  完成 下发任务:9 当前线程总数:5 队列中的线程数量:3
1655083181101  MyTask [name=10] rejected from java.util.concurrent.ThreadPoolExecutor@4b1210ee[Running, pool size = 5, active threads = 5, queued tasks = 3, completed tasks = 0]
1655083181101  开始 下发任务:10 当前线程总数:5
1655083181101  完成 下发任务:10 当前线程总数:5 队列中的线程数量:3
1655083182085  MyTask [name=1] 运行结束! 运行线程 Thread-2
1655083182085  MyTask [name=3] 开始运行! 运行线程 Thread-2 需要(秒):1000
1655083182085 子线程 Thread-5【正在】监控主进程 Thread-2 任务3
1655083182095  MyTask [name=4] 开始运行! 运行线程 Thread-4 需要(秒):8000
1655083182095  MyTask [name=7] 运行结束! 运行线程 Thread-4
1655083182095 子线程 Thread-6【正在】监控主进程 Thread-4 任务4
1655083183090  MyTask [name=3] 运行结束! 运行线程 Thread-2
1655083183090  MyTask [name=5] 开始运行! 运行线程 Thread-2 需要(秒):1000
1655083183090 子线程 Thread-7【正在】监控主进程 Thread-2 任务5
1655083184100  MyTask [name=5] 运行结束! 运行线程 Thread-2
1655083185087  MyTask [name=2] 运行结束! 运行线程 Thread-1
1655083186085 子线程 Thread-0【完成】监控主进程 Thread-2 任务1
1655083186086 子线程 Thread-1【完成】监控主进程 Thread-1 任务2
1655083186094  MyTask [name=6] 被打断 运行线程 Thread-3
1655083186094 子线程 Thread-2【杀掉】主进程(超时了) Thread-3 任务6
1655083186094 子线程 Thread-2【完成】监控主进程 Thread-3 任务6
1655083186096 子线程 Thread-3【完成】监控主进程 Thread-4 任务7
1655083186098  MyTask [name=8] 被打断 运行线程 Thread-5
1655083186098 子线程 Thread-4【杀掉】主进程(超时了) Thread-5 任务8
1655083186098 子线程 Thread-4【完成】监控主进程 Thread-5 任务8
1655083187087 子线程 Thread-5【完成】监控主进程 Thread-2 任务3
1655083187103  MyTask [name=4] 被打断 运行线程 Thread-4
1655083187103 子线程 Thread-6【杀掉】主进程(超时了) Thread-4 任务4
1655083187103 子线程 Thread-6【完成】监控主进程 Thread-4 任务4
1655083188098 子线程 Thread-7【完成】监控主进程 Thread-2 任务5
预启动线程(备战)
预启动线程数(备战):2
目录
相关文章
|
5月前
|
监控 Java 调度
Java线程池ThreadPoolExecutor初略探索
Java线程池ThreadPoolExecutor初略探索
|
6月前
|
数据采集 存储 Java
「多线程大杀器」Python并发编程利器:ThreadPoolExecutor,让你一次性轻松开启多个线程,秒杀大量任务!
「多线程大杀器」Python并发编程利器:ThreadPoolExecutor,让你一次性轻松开启多个线程,秒杀大量任务!
|
6月前
|
数据采集 Java Python
python并发编程: Python好用的线程池ThreadPoolExecutor
python并发编程: Python好用的线程池ThreadPoolExecutor
123 2
python并发编程: Python好用的线程池ThreadPoolExecutor
|
5月前
|
监控 安全 Java
深入理解Java线程池:ThreadPoolExecutor
深入理解Java线程池:ThreadPoolExecutor
68 0
|
5月前
|
Java
线程池ThreadPoolExecutor总结
线程池ThreadPoolExecutor总结
|
6月前
|
Dubbo 安全 Java
ThreadPoolExecutor线程池参数及其设置规则
ThreadPoolExecutor线程池参数及其设置规则
70 1
|
缓存 Java p3c
【Java用法】线程池不允许使用Executors去创建,而是通过ThreadPoolExecutor的方式,这样的处理方式让写的同学更加明确线程池的运行规则,规避资源耗尽的风险。
【Java用法】线程池不允许使用Executors去创建,而是通过ThreadPoolExecutor的方式,这样的处理方式让写的同学更加明确线程池的运行规则,规避资源耗尽的风险。
182 0
|
6月前
|
并行计算 安全 Java
Python中如何使用ThreadPoolExecutor一次开启多个线程
Python中如何使用ThreadPoolExecutor一次开启多个线程
108 0
|
11月前
|
消息中间件 缓存 Dubbo
java线程池ThreadPoolExecutor八种拒绝策略浅析
java线程池ThreadPoolExecutor八种拒绝策略浅析
273 0
|
6月前
|
Java
Java【代码分享 10】线程池ThreadPoolExecutor指定线程执行任务(修改线程名称+线程任务指定)
Java【代码分享 10】线程池ThreadPoolExecutor指定线程执行任务(修改线程名称+线程任务指定)
210 0