博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Java实现终止线程池中正在运行的定时任务
阅读量:7113 次
发布时间:2019-06-28

本文共 5015 字,大约阅读时间需要 16 分钟。

源于开发

最近项目中遇到了一个新的需求,就是实现一个可以动态添加定时任务的功能。说到这里,有人可能会说简单啊,使用quartz就好了,简单粗暴。然而quartz框架太重了,小项目根本不好操作啊。当然,也有人会说,jdk提供了timer的接口啊,完全够用啊。但是我们项目的需求完全是多线程的模型啊,而timer是单线程的,so,楼主最后还是选择了jdk的线程池。

线程池是什么

Java通过Executors提供四种线程池,分别为: **newCachedThreadPool :**创建一个可缓存线程池,如果线程池长度超过处理需要,可灵活回收空闲线程,若无可回收,则新建线程。 newFixedThreadPool : 创建一个定长线程池,可控制线程最大并发数,超出的线程会在队列中等待。 newScheduledThreadPool : 创建一个定长线程池,支持定时及周期性任务执行。 newSingleThreadExecutor : 创建一个单线程化的线程池,它只会用唯一的工作线程来执行任务,保证所有任务按照指定顺序(FIFO, LIFO, 优先级)执行。

楼主项目中用到的是newScheduledThreadPool, 就这些吧,再多的楼主就班门弄斧了,Google一下,一大堆。

线程池service的获取

楼主通过单例模式来获取线程池的service,代码如下:

/** * 线程池创建. * @author wuhf * @date 2018/01/16 */public class ThreadPoolUtils {    private static ScheduledExecutorService executorService;    private ThreadPoolUtils() {        //手动创建线程池.        executorService = new ScheduledThreadPoolExecutor(10,                new BasicThreadFactory.Builder().namingPattern("syncdata-schedule-pool-%d").daemon(true).build());    }    private static class PluginConfigHolder {        private final static ThreadPoolUtils INSTANCE = new ThreadPoolUtils();    }    public static ThreadPoolUtils getInstance() {        return PluginConfigHolder.INSTANCE;    }    public ScheduledExecutorService getThreadPool(){        return executorService;    }}复制代码

中断某一个正在运行的线程代码实现

废话就不多说了,代码如下:

/** * 中断线程池的某个任务. */public class InterruptThread implements Runnable {    private int num;    public InterruptThread (int num){        this.num = num;    }    public static void main(String[] args) throws InterruptedException {        Thread interruptThread = new Thread(new InterruptThread(1));        ScheduledFuture
t = ThreadPoolUtils.getInstance().getThreadPool().scheduleAtFixedRate(interruptThread,0,2, TimeUnit.SECONDS); InterruptThread interruptThread1 = new InterruptThread(2); ThreadPoolUtils.getInstance().getThreadPool().scheduleAtFixedRate(interruptThread1,0,2, TimeUnit.SECONDS); InterruptThread interruptThread2 = new InterruptThread(3); ThreadPoolUtils.getInstance().getThreadPool().scheduleAtFixedRate(interruptThread2,0,2, TimeUnit.SECONDS); Thread.sleep(5000); //终止正在运行的线程interruptThread t.cancel(true); while (true){ } } @Override public void run() { System.out.println("this is a thread" + num); }}复制代码

踩坑记录

楼主在使用如下代码时,突然想到当这个定时任务需要被停止时该如何停止线程运行

ThreadPoolUtils.getInstance().getThreadPool().scheduleAtFixedRate(interruptThread,0,2,                TimeUnit.SECONDS);复制代码

既然我有这样的需求,那就Google一下吧,找了大半圈,愣是没找到相关资料,都是一些关于Java线程池的深入分析。或者是全局变量啥的,并没有找到令楼主满意的解决方案。

既然没有线程的那就扒一下scheduleAtFixedRate的底层源码看看是什么东西吧,果不其然我在源码中看到了scheduleAtFixedRate方法的具体实现,发现他的返回值是ScheduledFuture。

public ScheduledFuture
scheduleAtFixedRate(Runnable command, long initialDelay, long period, TimeUnit unit) { if (command == null || unit == null) throw new NullPointerException(); if (period <= 0) throw new IllegalArgumentException(); ScheduledFutureTask
sft = new ScheduledFutureTask
(command, null, triggerTime(initialDelay, unit), unit.toNanos(period)); RunnableScheduledFuture
t = decorateTask(command, sft); sft.outerTask = t; delayedExecute(t); return t; }复制代码

接着往下我们再看看ScheduledFuture里面有什么东西吧,没有让楼主失望,看到了这个

public boolean cancel(boolean mayInterruptIfRunning) {            boolean cancelled = super.cancel(mayInterruptIfRunning);            if (cancelled && removeOnCancel && heapIndex >= 0)                remove(this);            return cancelled;}            //从线程的运行队列中移除当前线程public boolean remove(Runnable task) {        boolean removed = workQueue.remove(task);        tryTerminate(); // In case SHUTDOWN and now empty        return removed;}复制代码

再往上查super.cancel(mayInterruptIfRunning)是什么东西,我们看到了这个,

//通过调用线程的interrupt方法终止线程运行public boolean cancel(boolean mayInterruptIfRunning) {        if (!(state == NEW &&              UNSAFE.compareAndSwapInt(this, stateOffset, NEW,                  mayInterruptIfRunning ? INTERRUPTING : CANCELLED)))            return false;        try {    // in case call to interrupt throws exception            if (mayInterruptIfRunning) {                try {                    Thread t = runner;                    if (t != null)                        t.interrupt();                } finally { // final state                    UNSAFE.putOrderedInt(this, stateOffset, INTERRUPTED);                }            }        } finally {            finishCompletion();        }        return true;    }复制代码

到这里所有的问题都迎刃而解。

总结一下吧

项目中总是会遇到比较难搞的解决方案,当Google不太好找时,翻一下jdk的源码或许也是一个不错的方法。最后宣传一下楼主的博客,博客已经迁移到腾讯云上,跟几个大学的小伙伴一起经营,内容涵盖java,Android等,感兴趣的小伙伴请移步

转载地址:http://ubwel.baihongyu.com/

你可能感兴趣的文章
DirectUI: Become windowless
查看>>
Python 数据结构_队列
查看>>
NAS数据迁移初探
查看>>
打破医院围墙 数字化平台之上的想象力
查看>>
Teradata首席分析官Bill Franks:数据分析变革犹如一场工业革命
查看>>
Linux下安装并使用Java开发opencv的配置
查看>>
AdTime: DMC量身定制的企业数据分析师
查看>>
《数字逻辑设计与计算机组成》一2.3 规范表达式
查看>>
借道大数据 互联网基金再探“蓝海”
查看>>
浙江医疗综合体获批 医疗资源也可共享
查看>>
3G/4G调制解调器曝漏洞:可致设备被完全控制
查看>>
你知道你的Mac摄像头正在偷窥你吗?这款工具或许能帮你
查看>>
超干货!一套完整的设计分析思路应该是怎样的?
查看>>
一文深入了解Redis!
查看>>
js判断document.getElementByid("")获得的对象是否存在
查看>>
实用的 Recipes
查看>>
从最大似然到EM算法浅解
查看>>
ionic 拍照、相册并上传至又拍云
查看>>
NIO 理解
查看>>
第十三章:通过UDP广播事件
查看>>