疯狂java


您现在的位置: 疯狂软件 >> 新闻资讯 >> 正文

定时任务Java编写综述之Quartz实战


 

  4.2 Eclipse 中的 Quartz 工程

  创建一个 Quartz Job 类

  每一个 Quartz Job 必须有一个实现了 org.quartz.Job 接口的具体类。这个接口仅有一个实现的方法,即execute(),方法 execute() 的原型如下:

  public void execute(JobExecutionContext context) throws JobExecutionException;

  当 Quartz 调度器确定到时间要激发一个 Job 的时候,它就会生成一个 Job 实例,并调用这个实例的 execute() 方法。调度器只管调用 execute() 方法,而不关心执行的结果,除了在作业执行中出问题抛出的 org.quartz.JobExecutionException 异常。

  因此,用户可以在 execute() 方法中执行相应的业务逻辑。

  当 Quartz 调用 execute() 方法,会传递一个 org.quartz.JobExecutionContext 上下文变量,里面封装有 Quartz 的运行时环境和当前正执行的 Job。通过 JobexecutionContext,可以访问到调度器的信息、作业和作业上的触发器的信息、还有更多更多的信息。

  JobDetail 持有一个指向 org.quartz.JobDataMap 的引用。JobDataMap 中有为指定 Job 配置的自定义属性。

  以下通过实例展示:

  private void autoFuction(long fixedTime) throws SchedulerException {

  SchedulerFactory schedulerFactory = new StdSchedulerFactory();

  Scheduler scheduler = schedulerFactory.getScheduler();

  JobDetail jobDetail = new JobDetail("autoFuction ", " autoFuctionGroup", autoFuctionJob.class);

  DateTime beginTime = new DateTime(DateTime.current(), DateTime.YEAR_TO_MILLISECOND);

  beginTime = beginTime.addMinute(10);

  System.out.println("定时任务开始时间:" + beginTime + ",周期" + fixedTime + "S");

  SimpleTrigger simpleTrigger = new SimpleTrigger("autoFuctionTrigger", "a autoFuction riggerGroup", beginTime);

  simpleTrigger.setRepeatInterval(fixedTime * 1000);

  simpleTrigger.setRepeatCount(-1);

  scheduler.scheduleJob(jobDetail, simpleTrigger);

  scheduler.start();

  }

  public class autoFuctionJob implements Job {

  public void execute(JobExecutionContext jobContext) throws JobExecutionException {

  try {

  //do something

  } catch(Exception e){

  e.printStackTrace();

  }

  }

  }

  从上述实例可以看到,Quartz Schduler 是框架的心脏与灵魂。所有的 Job 都通过 Schduler 注册。

  Quartz 的触发器主要有CronTrigger和SimpleTrigger。上例已经展示了SimpleTrigger的用法,SimpleTrigger对于需要在指定的毫秒处及时执行的作业还是不错的,但是假如你的作业需要更复杂的执行计划时,你也就要 CronTrigger给你提供更强更灵活的功能。

  crontab 格式包含六段,前五段为执行计划,第六段为要执行的命令。(Quartz cron 表达式有七段。) 下面这些是执行计划的五个字段:

  ·分 (00-59)

  ·时 (00-23)

  ·日 (1-31)

  ·月 (1-12)

  ·周 (0-6 或 sun-sat)

  UNIX cron 格式表达式中允许出现一些特殊的字符,例如星号(*),它用来匹配所有值。

  用法如下:

  try {

  scheduler = StdSchedulerFactory.getDefaultScheduler();

  scheduler.start();

  JobDetail jobDetail = new JobDetail("PrintInfoJob", Scheduler.DEFAULT_GROUP,

  PrintInfoJob.class);

  try {

  CronTrigger trigger = new

  CronTrigger("CronTrigger", null,

  "0 30 7 ? * MON-FRI");

  scheduler.scheduleJob(jobDetail, trigger);

  } catch (ParseException ex) {

  logger.error("Error parsing cron expr", ex);

  }

  } catch (SchedulerException ex) {

  logger.error(ex);

  }

  Quartz Cron 表达式支持到七个域 名称是否必须允许值特殊字符

  秒是 0-59 , - * /

  分是 0-59 , - * /

  时是 0-23 , - * /

  日是 1-31 , - * ? / L W C

  月是 1-12 或 JAN-DEC , - * /

  周是 1-7 或 SUN-SAT , - * ? / L C #

  年否空或 1970-2099 , - * /

  月份和星期的名称是不区分大小写的。FRI 和 fri 是一样的。

  理解特殊字符

  同 UNIX cron 一样,Quartz cron 表达式支持用特殊字符来创建更为复杂的执行计划。然而,Quartz 在特殊字符的支持上比标准 UNIX cron 表达式更丰富了。

  * 星号

  使用星号(*) 指示着你想在这个域上包含所有合法的值。例如,在月份域上使用星号意味着每个月都会触发这个 trigger。

  表达式样例:

  0 * 17 * * ?

  意义:每天从下午5点到下午5:59中的每分钟激发一次 trigger。它停在下午 5:59 是因为值 17 在小时域上,在下午 6 点时,小时变为 18 了,也就不再理会这个 trigger,直到下一天的下午5点。在你希望 trigger 在该域的所有有效值上被激发时使用 * 字符。

  ? 问号

  ? 号只能用在日和周域上,但是不能在这两个域上同时使用。你可以认为 ? 字符是 "我并不关心在该域上是什么值。" 这不同于星号,星号是指示着该域上的每一个值。? 是说不为该域指定值。不能同时这两个域上指定值的理由是难以解释甚至是难以理解的。基本上,假定同时指定值的话,意义就会变得含混不清了:考虑一下,如果一个表达式在日域上有值11,同时在周域上指定了 WED。那么是要 trigger 仅在每个月的11号,且正好又是星期三那天被激发?还是在每个星期三的11号被激发呢?要去除这种不明确性的办法就是不能同时在这两个域上指定值。

  只要记住,假如你为这两域的其中一个指定了值,那就必须在另一个字值上放一个 ?。

  表达式样例:

  0 10,44 14 ? 3 WEB

  意义:在三月中的每个星期三的下午 2:10 和下午 2:44 被触发。

  , 逗号

  逗号 (,) 是用来在给某个域上指定一个值列表的。例如,使用值 0,15,30,45 在秒域上意味着每15秒触发一个 trigger。

  表达式样例:

  0 0,15,30,45 * * * ?

  意义:每刻钟触发一次 trigger。

  / 斜杠

  斜杠 (/) 是用于时间表的递增的。我们刚刚用了逗号来表示每15分钟的递增,但是我们也能写成这样 0/15。

  表达式样例:

  0/15 0/30 * * * ?

  意义:在整点和半点时每15秒触发 trigger。

  - 中划线

  中划线 (-) 用于指定一个范围。例如,在小时域上的 3-8 意味着 "3,4,5,6,7 和 8 点。" 域的值不允许回卷,所以像 50-10 这样的值是不允许的。

  表达式样例:

  0 45 3-8 ? * *

  意义:在上午的3点至上午的8点的45分时触发 trigger。

  L 字母

  L 说明了某域上允许的最后一个值。它仅被日和周域支持。当用在日域上,表示的是在月域上指定的月份的最后一天。例如,当月域上指定了 JAN 时,在日域上的 L 会促使 trigger 在1月

  31号被触发。假如月域上是 SEP,那么 L 会预示着在9月30号触发。换句话说,就是不管指定了哪个月,都是在相应月份的时最后一天触发 trigger。

  表达式 0 0 8 L * ? 意义是在每个月最后一天的上午 8:00 触发 trigger。在月域上的 * 说明是 "每个月"。

  当 L 字母用于周域上,指示着周的最后一天,就是星期六 (或者数字7)。所以如果你需要在每个月的最后一个星期六下午的 11:59 触发 trigger,你可以用这样的表达式 0 59 23 ? * L。

  当使用于周域上,你可以用一个数字与 L 连起来表示月份的最后一个星期 X。例如,表达式 0 0 12 ? * 2L 说的是在每个月的最后一个星期一触发 trigger。

  不要让范围和列表值与 L 连用

  虽然你能用星期数(1-7)与 L 连用,但是不允许你用一个范围值和列表值与 L 连用。这会产生不可预知的结果。

  W 字母

  W 字符代表着平日 (Mon-Fri),并且仅能用于日域中。它用来指定离指定日的最近的一个平日。大部分的商业处理都是基于工作周的,所以 W 字符可能是非常重要的。例如,日域中的 15W 意味着 "离该月15号的最近一个平日。" 假如15号是星期六,那么 trigger 会在14号(星期四)触发,因为距15号最近的是星期一,这个例子中也会是17号(译者Unmi注:不会在17号触发的,如果是15W,可能会是在14号(15号是星期六)或者15号(15号是星期天)触发,也就是只能出现在邻近的一天,如果15号当天为平日直接就会当日执行)。W 只能用在指定的日域为单天,不能是范围或列表值。

  # 井号

  # 字符仅能用于周域中。它用于指定月份中的第几周的哪一天。例如,如果你指定周域的值为 6#3,它意思是某月的第三个周五 (6=星期五,#3意味着月份中的第三周)。另一个例子 2#1 意思是某月的第一个星期一 (2=星期一,#1意味着月份中的第一周)。注意,假如你指定 #5,然而月份中没有第 5 周,那么该月不会触发。

  还可以用 CronTrigger的 setStartTime()和 setEndTime()方法来形成一个 "定时箱" 来触发。

  疯狂软件学院是专业做软件开发培训的,主要课程分为Java和Android课程。疯狂Java团队大部分都是项目经理、技术总监,并且部分老师还是疯狂Java体系图书的作者。疯狂软件Java打造名企技术经理,成为中国软件产业的中流砥柱:全真企业需求,大量实操项目的疯狂训练,企业全真案例教学,高强度的狂热训练,带领学生走近编程,感受编程,热爱编程,充分激发每个程序员(学员)对编程的激情四个半月掌握近8~10万代码量,达到技术经理的代码掌握量,学成至少相当于两年工作经验,实战拥有了多种项目经验,成为可以立刻上手实际工作的全能型IT人才。疯狂Java培训抢座热线:020-28309358,020-28309378咨询QQ:707552864,544627560