Quartz是一个开源的作业调度框架,它完全由Java写成,并设计用于J2SE和J2EE应用中。它提供了巨 大的灵 活性而不牺牲简单性。你能够用它来为执行一个作业而创建简单的或复杂的调度。它有很多特征,如:数据库支持,集群,插件,EJB作业预构 建,JavaMail及其它,支持cron-like表达式等等。
该项目已经被 Terracotta收购。
Quartz使用类似于Linux下的Cron表达式定义时间规则,Cron表达式由6或7个由空格分隔的时间字段组成,如表1所示:
表 1 Cron表达式时间字段
Cron表达式的时间字段除允许设置数值外,还可使用一些特殊的字符,提供列表、范围、通配符等功能,细说如下:
●星号(*):可用在所有字段中,表示对应时间域的每一个时刻,例如,*在分钟字段时,表示“每分钟”;
●问号(?):该字符只在日期和星期字段中使用,它通常指定为“无意义的值”,相当于点位符;
●减号(-):表达一个范围,如在小时字段中使用“10-12”,则表示从10到12点,即10,11,12;
●逗号(,):表达一个列表值,如在星期字段中使用“MON,WED,FRI”,则表示星期一,星期三和星期五;
●斜杠(/):x/y表达一个等步长序列,x为起始值,y为增量步长值。如在分钟字段中使用0/15,则表示为0,15,30和45秒,而5/15在分钟字段中表示5,20,35,50,你也可以使用*/y,它等同于0/y;
●L:该字符只在日期和星期字段中使用,代表“Last”的意思,但它在两个字段中意思不同。L在日期字段中,表示这个 月份的最后一天,如一月的31号,非闰年二月的28号;如果L用在星期中,则表示星期六,等同于7。但是,如果L出现在星期字段里,而且在前面有一个数值 X,则表示“这个月的最后X天”,例如,6L表示该月的最后星期五;
●W:该字符只能出现在日期字段里,是对前导日期的修饰,表示离该日期最近的工作日。例如15W表示离该月15号最近的 工作日,如果该月15号是星期六,则匹配14号星期五;如果15日是星期日,则匹配16号星期一;如果15号是星期二,那结果就是15号星期二。但必须注 意关联的匹配日期不能够跨月,如你指定1W,如果1号是星期六,结果匹配的是3号星期一,而非上个月最后的那天。W字符串只能指定单一日期,而不能指定日 期范围;
●LW组合:在日期字段可以组合使用LW,它的意思是当月的最后一个工作日;
●井号(#):该字符只能在星期字段中使用,表示当月某个工作日。如6#3表示当月的第三个星期五(6表示星期五,#3表示当前的第三个),而4#5表示当月的第五个星期三,假设当月没有第五个星期三,忽略不触发;
● C:该字符只在日期和星期字段中使用,代表“Calendar”的意思。它的意思是计划所关联的日期,如果日期没有被关联,则相当于日历中所有日期。例如5C在日期字段中就相当于日历5日以后的第一天。1C在星期字段中相当于星期日后的第一天。
Cron表达式对特殊字符的大小写不敏感,对代表星期的缩写英文大小写也不敏感。
表2下面给出一些完整的Cron表示式的实例:
package job; import java.text.ParseException; import java.util.ArrayList; import java.util.Date; import org.quartz.CronTrigger; import org.quartz.JobDetail; import org.quartz.Scheduler; import org.quartz.SchedulerException; import org.quartz.SchedulerFactory; import org.quartz.SchedulerMetaData; import org.quartz.SimpleTrigger; import org.quartz.Trigger; import org.quartz.TriggerUtils; import org.quartz.impl.StdSchedulerFactory; public class TimerTest { public static void main(String[] string) throws ParseException { SchedulerFactory schedFact = new org.quartz.impl.StdSchedulerFactory(); Scheduler sched; try { //创建调度 // sched = schedFact.getScheduler(); sched = StdSchedulerFactory.getDefaultScheduler(); //创建任务 JobDetail jobDetail = new JobDetail("myJob", // job name sched.DEFAULT_GROUP, // job group (you can also specify 'null' to use the default group) Job1.class); // the java class to execute //传递的参数 jobDetail.getJobDataMap().put("who", "张三"); jobDetail.getJobDataMap().put("content", "你好!"); jobDetail.getJobDataMap().put("myFloatValue", 3.141f); jobDetail.getJobDataMap().put("myStateData", new ArrayList()); //表达式cron触发器(比较强悍的) CronTrigger cronTrigger = new CronTrigger(); cronTrigger.setCronExpression("0/3 10-59 * * * ?"); cronTrigger.setName("tri1"); cronTrigger.setGroup("test"); sched.scheduleJob(jobDetail, cronTrigger); //再添加一个任务 JobDetail job2 = new JobDetail("myJob2", // job name sched.DEFAULT_GROUP, // job group (you can also specify 'null' to use the default group) Job2.class); // the java class to execute CronTrigger cronTrigger2 = new CronTrigger(); cronTrigger2.setCronExpression("0/3 10-59 * * * ?");//从0秒开始每隔3秒、从第20到59分钟、所有小时、所有日、所有月、所有周、所有年(可略) cronTrigger2.setName("test002"); cronTrigger2.setGroup("test"); //CronTrigger cronTrigger2 = new CronTrigger("trigger1", "group1", "job2", "group1", "0/20 * * * * ?"); sched.scheduleJob(job2, cronTrigger2); //简单触发器 /*SimpleTrigger simpleTrigger = new SimpleTrigger(); simpleTrigger.setStartTime(new Date()); simpleTrigger.setName("test001"); cronTrigger.setGroup("test"); sched.scheduleJob(jobDetail, simpleTrigger);*/ //按秒触发器 // Trigger trigger = TriggerUtils.makeSecondlyTrigger(3);//每隔3秒 // trigger.setStartTime(new Date()); // trigger.setName("test001"); // cronTrigger.setGroup("test"); // sched.scheduleJob(jobDetail, trigger); //按天触发器 // Trigger trigger = TriggerUtils.makeDailyTrigger(0, 1); // trigger.setStartTime(new Date()); // trigger.setName("myTrigger"); // cronTrigger.setGroup("test"); // sched.scheduleJob(jobDetail, trigger); //启动调度任务 sched.start(); // sched.pauseAll();//暂停所有任务 // scheduler.deleteJob(String jobName, String groupName);//删除制定job // scheduler.resumeAll();//恢复所有任务 //结束调度任务 //sched.shutdown(); } catch (SchedulerException e) { e.printStackTrace(); } } }
package job; import java.util.ArrayList; import java.util.Date; import org.quartz.Job; import org.quartz.JobDataMap; import org.quartz.JobExecutionContext; import org.quartz.JobExecutionException; public class Job1 implements Job { public void execute(JobExecutionContext context) throws JobExecutionException { System.out.println("........................................"); String instName = context.getJobDetail().getName(); String instGroup = context.getJobDetail().getGroup(); JobDataMap dataMap = context.getJobDetail().getJobDataMap(); String who = dataMap.getString("who"); String jobSays = dataMap.getString("content"); float myFloatValue = dataMap.getFloat("myFloatValue"); ArrayList state = (ArrayList) dataMap.get("myStateData"); state.add(new Date()); System.out.println("Instance " + instName + " of DumbJob ,"+who+" says: " + jobSays); } }
package job; import java.util.ArrayList; import java.util.Date; import org.quartz.Job; import org.quartz.JobDataMap; import org.quartz.JobDetail; import org.quartz.JobExecutionContext; import org.quartz.JobExecutionException; import org.quartz.SchedulerException; import org.quartz.SchedulerMetaData; public class Job2 implements Job { public void execute(JobExecutionContext context) throws JobExecutionException { String instName = context.getJobDetail().getName(); String instGroup = context.getJobDetail().getGroup(); JobDataMap dataMap = context.getJobDetail().getJobDataMap(); try { /*JobDetail job = context.getScheduler().getJobDetail("test002", "test"); System.out.println(job.getFullName());*/ SchedulerMetaData metaData = context.getScheduler().getMetaData(); System.out.println("Executed " + metaData.numJobsExecuted() + " jobs."); } catch (SchedulerException e) { e.printStackTrace(); } System.out.println("这是任务2!"); } }
org.quartz.scheduler.instanceName = DefaultQuartzScheduler org.quartz.scheduler.rmi.export = false org.quartz.scheduler.rmi.proxy = false org.quartz.scheduler.wrapJobExecutionInUserTransaction = false org.quartz.threadPool.class = org.quartz.simpl.SimpleThreadPool org.quartz.threadPool.threadCount = 10 org.quartz.threadPool.threadPriority = 5 org.quartz.threadPool.threadsInheritContextClassLoaderOfInitializingThread = true org.quartz.jobStore.misfireThreshold = 60000 #org.quartz.jobStore.class = org.quartz.simpl.RAMJobStore #内存保存 org.quartz.jobStore.class = org.quartz.simpl.RAMJobStore #持久化到数据库 #org.quartz.jobStore.class = org.quartz.impl.jdbcjobstore.JobStoreTX #org.quartz.jobStore.driverDelegateClass=org.quartz.impl.jdbcjobstore.StdJDBCDelegate #org.quartz.jobStore.useProperties = true #org.quartz.jobStore.tablePrefix = QRTZ_ #org.quartz.jobStore.isClustered = false #org.quartz.jobStore.maxMisfiresToHandleAtATime=1