一、什么是Quartz?
Quartz是已给完全由Java编写的开源作业调度框架,为在Java应用程序中进行作业调度提供了简单却强大的机制。
Quartz是OpenSymphony开源组织Job scheduling领域又一个开源项目,它可以与J2EE和J2SE应用相结合也可以单独使用。
二、Quartz的作用
Quartz定义一个触发条件,到时间点后触发响应的Job起来干活。
在某一个有规律的时间点干某件事。很容易定义触发条件。
三、Quartz核心概念
1、Job(任务):定义具体执行的任务逻辑(内容);
2、JobDetail(任务详情):定义一个可执行的调度程序,Job是这个可执行调度程序所要执行的内容;
3、Trigger(任务触发器):定义任务执行的方式、间隔;
4、Scheduler(任务调度器):控制所有的任务调度(Trigger与JobDetail组合)。
注:任务是有可能并发执行,如果Scheduler直接使用Job,就会存在对同一个Job实例并发访问的问题。而JobDetail和Job组合方式,Sheduler每次执行都会根据JobDetail创建一个新的Job实例,这样就可以规避并发访问的问题。
四、导入依赖包
<!--导入Quartz依赖包/可以在官网下载最新版 -->
<dependency>
<groupId>org.quartz-scheduler</groupId>
<artifactId>quartz</artifactId>
<version>2.2.1</version>
</dependency>
<dependency>
<groupId>org.quartz-scheduler</groupId>
<artifactId>quartz-jobs</artifactId>
<version>2.2.1</version>
</dependency>
五、简单Quartz应用-编写步骤
1、自定义Job类,实现Job接口的execute()方法;
2、创建JobDetail实例,定义Job名称和参数;
3、创建Trigger实例,定义Job执行的间隔时间;
4、创建Scheduler实例,添加JobDetail和Trigger,启动任务调度。
六、Trigger四种实现方式
1、SimpleTrigger
指定从某一时间开始,以一定的时间间隔(单位毫秒)执行的任务;它适合的任务类似于:9:00开始,每隔1小时执行一次。
属性:repeatInterval-重复间隔、repeatCount-重复次数
//举例1
sumpleScheduule()
.withIntervalInHours(1) //每小时执行一次
.repeatForever()//次数不限
.build();
//举例2
sumpleScheduule()
.withIntervalInMinutes(1) //每分钟执行一次
.withRepeatCount(10)//次数为10次
.build();
2、CalendarIntervalTrigger
类似于SimpleTrigger,指定从某一个时间开始,以一定的时间间隔执行的任务;CalendarIntervalTrigger支持的间隔单位有秒、分钟、小时、天、月、年、星期。它适合的任务类似于:9:00开始执行,并且以后每周9:00执行一次。
优势:①更方便(比如每隔1小时执行,你不用自己去计算1小时等于多少毫秒)
②支持不是固定长度的间隔(比如间隔为月和年)
劣势:只能精确到秒。
属性:interval-执行间隔、intervalUnit-执行间隔的单位(秒、分钟、小时、天、月、年、星期)
//举例1
calendarIntervalScheduule()
.withIntervalInDays(1) //每天执行一次
.build();
//举例2
calendarIntervalScheduule()
.withIntervalInWeeks(1) //每周执行一次
.build();
3、DailyTimeIntervalTrigger
指定每天的某个时间段内,以一定的时间间隔执行任务。并且它可以支持指定星期。它适合的任务类似于:指定每天9:00至18:00,每隔70秒执行一次,并且只要周一至周五执行。可以满足更为复杂的需求。
属性:1)startTimeOfDay-每天开始时间
2)endTimeOfDay-每天结束时间
3)daysOfWeek-需要执行的星期
4)intervall-执行间隔
5)intervalUnit-执行间隔的单位(秒、分钟、小时、天、月、年、星期)
6)repeatCount-重复次数
4、CronTrigger
适合于更复杂的任务,它支持类型与Linux Cron的语法(并且更强大),基本上它覆盖了以上三个Trigger的绝大部分能力(但不是全部);当然,也更难理解和掌握。它适合的任务类似于:每天0:00,9:00,18:00各执行一次。
属性:Cron表达式
cronScheduule("0 0/2 8-17 * * ?")//每天8:00-17:00,每隔2分钟执行一次
.build();
cronScheduule("0 30 9 ? * MON")//每周一,9:30执行一次
.build();
5、Scheduler和SchedulerFactory
Scheduler就是Quartz的大脑,所有任务都是由它来控制调度,它包含两个重要组件:JobStore和ThreadPool。
JobStore-是会来存储运行时信息的,包括Trigger、Schduler、JobDetail、业务锁等。它有多种实现RAMJob(内存实现),JobStoreTX(JDBC,事务由Quartz管理),JobStoreCMT(JDBC,使用容器事务),ClusteredJobStore(集群实现)、TerracottaJobStore
ThreadPool-就是线程池,Quartz有自己的线程池实现;所有任务的都会由线程池执行。
SchedulerFactory-顾名思义就是用来创建Schduler了,有两个实现:DirectSchedulerFactory和StdSchedulerFactory,通常来讲,我们使用StdSchedulerFactory也就足够了。
注:SchedulerFactory本身是支持创建RMI stub的,可以用来管理远程的Scheduler,功能与本地一样,可以远程提交Job。
Quartz扩展:集群、插件、监听器、RMI、JobStore
特殊字符 | 意义 |
* | 匹配所有的值。如:*在分钟的字段域里表示 每分钟 |
? | 只在日期域和星期域中使用。它被用来指定“非明确的值” |
- | 指定一个范围。如:“10-12”在小时域意味着“10点、11点、12点” |
, | 指定几个可选值。如:“MON,WED,FRI”在星期域里表示“星期一、星期三、星期五” |
/ | 指定增量。如:“0/15”在秒域意思是每分钟的0,15,30和45秒。“5/15”在分钟域表示没小时的5,20,35和50。符号“*”在“/”前面(如:*/10)等价于0在“/”前面(如:0/10) |
L | 表示day-of-month和day-of-week域,但在两个字段中的意思不同,例如day-of-month域中表示一个月的最后一天。如果在day-of-week域表示‘7’或者‘SAT’,如果在day-of-week域中前面加上数字,它表示一个月的最后几天,例如‘6L’就表示一个月的最后一个星期五 |
W | 只允许日期域出现。这个字符用于指定日期的最近工作日。例如:如果你在日期域中写 “15W”,表示:这个月15号最近的工作日。所以,如果15号是周六,则任务会在14号触发。如果15好是周日,则任务会在周一也就是16号触发。如果是在日期域填写“1W”即使1号是周六,那么任务也只会在下周一,也就是3号触发,“W”字符指定的最近工作日是不能够跨月份的。字符“W”只能配合一个单独的数值使用,不能够是一个数字段,如:1-15W是错误的 |
LW | L和W可以在日期域中联合使用,LW表示这个月最后一周的工作日 |
# | 只允许在星期域中出现。这个字符用于指定本月的某某天。例如:“6#3”表示本月第三周的星期五(6表示星期五,3表示第三周)。“2#1”表示本月第一周的星期一。“4#5”表示第五周的星期三 |
C | 允许在日期域和星期域出现。这个字符依靠一个指定的“日历”。也就是说这个表达式的值依赖于相关的“日历”的计算结果,如果没有“日历”关联,则等价于所有包含的“日历”。如:日期域是“5C”表示关联“日历”中第一天,或者这个月开始的第一天的后5天。星期域是“1C”表示关联“日历”中第一天,或者星期的第一天的后1天,也就是周日的后一天(周一) |
表达式 | 解释 |
0 0 12 * * ? | 每天12点执行 |
0 15 10 ? * * | 每天10点15分执行 |
0 15 10 * * ? | 每天10点15分执行 |
0 15 10 * * ? * | 每年的每天10点15分执行 |
0 15 10 * * ? 2008 | 2008年的每天的10点15分执行 |
0 * 14 * * ? | 每天的14点每分钟执行 |
0 0/5 14 * * ? | 每天的14点每隔5分钟执行 |
0 0/5 14,18 * * ? | 每天的14点和18点每隔5分钟执行 |
0 0-5 14 * * ? | 每天的14点的0-5分钟执行 |
0 14,44 14 ? 3 WED | 3月的每周三的14点14分和44分执行 |
0 15 10 ? * MON-FRI | 每个月的周一到周五的10点15分执行 |
0 15 10 15 * ? | 每个月的15日的10点15分执行 |
0 15 10 L * ? | 每个月的最后一天的10点15分执行 |
0 15 10 ? * 6L | 每个月的最后一个星期五的10点15分执行 |
0 15 10 ? * 6L 2007-2009 | 2007-2009年的每个月的最后一个星期五的10点15分执行 |
0 15 10 ? * 6#3 | 每月的第三周的星期五的10点15分执行 |