三种基本的Java任务调度工具比较

1、Timer工具类

Timer是JDK自带的任务调度工具类,它只需要java.util.Timer和java.util.TimerTask两个类就可以实现基本任务调度功能。

其中TimerTask实现了Runnable接口的抽象类,开发人员只需要继承TimerTask并实现run方法,将任务内容写入run方法中,然后就可以将其交给Timer去调度执行了。

Timer类的调度方法有如下几种:

其中,task是需要被执行的任务,其它参数是调度计划的设置。

Timer类的核心是它的两个内部类TaskThread和TaskQueue。

在Timer创建的同时会初始化TaskThread和TaskQueue对象,并调用TaskThread的start方法启动该线程。Timer通过上述的schedule方法会将需要调度的TimerTask放入TaskQueue中,每次往TaskQueue放入新的TimerTask时, TaskQueue会按照任务的执行时间由小到大进行排序。

TimerThread线程在start方法启动后,就会开始不断轮询,每次轮询都会获取TaskQueue中第一个TimerTask( 执行时间最小的TimerTask),判断当前是否已到执行时间:

如当前时间大于或等于执行时间,则执行TimerTask;

如未到,则会休眠一段时间(时长=任务执行时间-当前时间)。

执行后,判定该task是否需要重复执行,如需要,则重置该task的执行时间,重新放入TaskQueue中(放入后会自动排序)。

Timer工具类

优点:JDK本身就自带该工具类,无需第三方依赖,只需实现TimerTask类即可使用Timer进行调度配置,使用起来简单方便。

缺点:Timer中所有的任务都一个TaskThread线程来调度和执行,任务的执行方式是串行的,如果前一个任务发生延迟或异常会影响到后续任务的执行。

2、ScheduledExecutorService

为了解决上述Timer缺陷,Sun公司在JDK1.5中引入了ScheduledThreadPoolExecutor类,它继承自ThreadPoolExecutor类并实现ScheduledExecutorService接口,故其能使用线程池来实现任务调度。

ScheduledExecutorService包含如下接口:

:scheduleAtFixedRate方法是基于初始延迟(initialDelay)后固定间隔(period)进行任务调度;scheduleWithFixedDelay方法是基于上次任务完成后固定的延迟时间来进行任务调度。两者的任务执行的维度不同。

ScheduledThreadPoolExecutor与Timer一样也有两个核心的内部类:DelayedWorkQueue和ScheduledFutureTask。

如上所示它的schedule方法接收一个Runnable接口的子类后,会将其包装成ScheduledFutureTask放入DelayedWorkQueue。DelayedWorkQueue是有序队列,会对新加入到队列的ScheduledFutureTask进行排序处理(按执行时间从小到大)。

而后DelayedWorkQueue中的任务会被调度线程从线程池中分配一个固定的线程进行调用,调用时执行run的方法会判断是否是周期性任务来决定是否要设置下次执行时间,以便下次执行。

ScheduledFutureTask的run方法如下:

ScheduledThreadPoolExecutor相对于Timer因为使用了线程池,所有任务都会单独分配一个线程去执行,故不会互相影响。在大部分情况下,推荐使用ScheduledThreadPoolExecutor来实现任务调度而非Timer。

ScheduledExecutorService虽然解决了Timer由于单线程导致的问题,但从上述schedule方法可以看出它是基于延迟(initialDelay)来设定具体执行时间的,虽然可以通过计算实现某些复杂的作业调度配置,但这种用法过于繁杂而且执行时间不够明确。

某些特殊的执行条件,如固定每个月几号执行,或是每天多个非固定间隔的时间点去执行同一任务等需求就无法实现,对于节假日等也无法控制,对此我们可以使用Quartz来实现这些相对复杂的调度需求。

3、Spring Quartz Scheduler

Spring Quartz是在Spring框架中使用Quartz工具来实现任务调度的方式,在Spring下对Quartz可以方便的完成任务调度需要的配置。

Quartz是一个相对上述两种调度工具更为复杂的任务调度系统,使用Trigger, Job 和JobDetail对象来实现对各种类型任务的调度。Spring也需要配置好这些对象,才能使用Quartz的任务调度功能。

Spring提供了一个JobDetailFactoryBean用于配置JobDetail,在JobDetail中包含了所有运行job (ExampleJob)需要的信息。让我们来看一下的例子:

其中ExampleJob是实现了Job接口的QuartzJobBean继承类,它是真正的任务执行者。具体实现如下所示:

Quartz的触发器是之前两种调度工具最大的区别,Quartz实现了两个常用的触发器SimpleTrigger和CronTrigger,SimpleTrigger可配置简单的执行计划。CronTrigger则可以根据具体的Corn表达式配置各种复杂的执行计划,满足各种特殊的需求。

Srping可以通过CronTriggerFactoryBean 和 SimpleTriggerFactoryBean来进行Trigger的配置。配置方法如下:

最后,只需要配置通过SchedulerFactoryBean来配置Scheduler,将Trigger注册到具体的Scheduler中,由其进行触发调度。

Spring即可以成功使用Quartz实现任务调度:

Quartz同ScheduledThreadPoolExecutor一样也是基于线程池进行任务调度的,它默认使用org.quartz.simpl.SimpleThreadPool来作为线程池,在调用scheduleJob()方法会将Job和Trigger存储在JobStore(从存储介质中获取触发器,存储介质可以是内存也可以是数据库)中,然后通知调度线程(QuartzSchedulerThread)从JobStore中获取即将被触发的触发器,到达触发时间后分配线程去执行触发器对应的Job任务。

本文作者:邱志辉(点融黑帮),目前就职于点融网工程部FinCore team,喜欢电影、旅行,以及一切新事物。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,242评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,769评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,484评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,133评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,007评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,080评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,496评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,190评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,464评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,549评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,330评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,205评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,567评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,889评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,160评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,475评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,650评论 2 335

推荐阅读更多精彩内容

  • 博客原文 徒手翻译spring framework 4.2.3官方文档的第33章,若有翻译不当之处请指正。 定时任...
    rabbitGYK阅读 5,603评论 4 24
  • 《分布式任务调度平台XXL-JOB》 一、简介 1.1 概述 XXL-JOB是一个轻量级分布式任务调度框架,其核心...
    许雪里阅读 16,749评论 3 29
  • 前言 在实际项目开发中,除了Web应用、SOA服务外,还有一类不可缺少的,那就是定时任务调度。定时任务的场景可以说...
    张丰哲阅读 29,751评论 15 57
  • Timer 定时器相信都不会陌生,之所以拿它来做源码分析,是发现整个控制流程可以体现很多有意思的东西。 在业务开发...
    石先阅读 6,288评论 2 13
  • 像我这种人,谁能受得了,对吧!每天总是妄想一些不可能的事,考研,支教,学法语,喜欢你,去希腊,看那个蓝白色的世界,...
    _夏天不冷阅读 171评论 0 0