:30发送email通知客户最新的业务情况。

java.util.Timer和java.util.TimerTask
    Timer和TimerTask是能够完毕job schedule的两个jdk提供的类,只是这不能称为一个system。Timer和TimerTask是非常easy的,不直接支持持久化任务,线程池和类似日历(calendar-like)的计划安排,在完毕一些高级功能上开发者要进行大量的扩展。

Quartz的简介
    Quartz是opensymphony组织专攻job scheduling领域又一个开源利器,能够到http://www.opensymphony.com/quartz查看具体信息。Quartz是轻量级的组件,开发者仅仅须要载入单独的jar包就能够利用Quartz强大的日程安排功能。当然,假如你为Quartz配备了数据库持久化任务的特性,Quartz也能够非常好的利用这一点。从而在机器重新启动后还能够记住你原先安排的计划。

    Quartz中我们接触最多的接口使Scheduler接口,该接口的提供了计划安排的功能。比方schedule/unschedule计划、start/pause/stop Scheduler.

    Quartz提供一些经常使用的Listener(JobListener,TriggerListener,SchedulerListener)用于全然的监视计划安排和运行情况。

開始我们的Quartz之旅

·  HelloWorld example:

想必大家非常想看一个HelloWorld的样例了吧。那么还是以HellowWorld開始。

import java.util.Date;
 import org.quartz.Job;
 import org.quartz.JobDetail;
 import org.quartz.JobExecutionContext;
 import org.quartz.JobExecutionException;
 import org.quartz.Scheduler;
 import org.quartz.SchedulerFactory;
 import org.quartz.Trigger;
 import org.quartz.impl.StdSchedulerFactory;
 /**
  * @author snowway
  * @version $Id$
  */
 public class SayHelloWorldJob implements Job{
     /*
      * (non-Javadoc)
      *
      * @see org.quartz.Job#execute(org.quartz.JobExecutionContext)
      */     public void execute(JobExecutionContext context) throws JobExecutionException{
         System.out.println("hello world!");
     }

     public static void main(String[] args) throws Exception{
         SchedulerFactory factory = new StdSchedulerFactory();
         Scheduler scheduler = factory.getScheduler();

         JobDetail jobDetail = new JobDetail("SayHelloWorldJob",
                 Scheduler.DEFAULT_GROUP,
                 SayHelloWorldJob.class);

         Trigger trigger = new SimpleTrigger("SayHelloWorldJobTrigger",
                 Scheduler.DEFAULT_GROUP,
                 new Date(),
                 null,
                 0,
                 0L);
         scheduler.scheduleJob(jobDetail, trigger);
         scheduler.start();
     }
 }

为了简单起见,我把main方法写在SayHelloWorldJob中了,运行SayHelloWorldJob能够看到控制台打印hello world.

·  回想Hello World example:

 Job是什么?
 接口Job是每一个业务上须要运行的任务须要实现的接口,该接口仅仅有一个方法:

 package org.quartz;

    public interface Job {

      public void execute(JobExecutionContext context)
        throws JobExecutionException;
    }

 execute方法也就是当时间到达后。Quartz回调的方法,我们使SayHelloWorldJob实现Job接口以提供打印功能

 JobDetail是什么? JobDetail描写叙述了一个任务具体的信息,比方名称,组名等等。

 JobDetail jobDetail = new JobDetail("SayHelloWorldJob",
                 Scheduler.DEFAULT_GROUP,
                 SayHelloWorldJob.class);
 在上面的构造方法中。第一个是任务的名称,第二个是组名,第三个就是实际当任务须要运行的回调类。

 Trigger是什么?
 Trigger顾名思义就是触发器。Quartz有个非常好的想法就是分离了任务和任务运行的条件。

Trigger就是控制任务运行条件的类。当Trigger觉得运行条件满足的时刻。Trigger会通知相关的Job去运行。分离的优点是:
 1.你能够为某个Job关联多个Trigger,当中不论什么一个条件满足都能够触发job运行,这样能够完毕一些组合的高级触发条件
 2.当Trigger失效后(比方:一个永远都不能满足的条件),你不必去声明一个新的job。取代的是你能够为job关联一个新的Trigger让job能够继续运行。

分钟后等等;而CronTrigger(没错,和unix的cron进程的含意一样)用来运行calendar-like的任务。比方:每周五下午3:00,每月最后一天等等。

 Trigger trigger = new SimpleTrigger("SayHelloWorldJobTrigger",
                 Scheduler.DEFAULT_GROUP,
                 new Date(),
                 null,
                 0,
                 0L); 这个构造方法中,第一个是Trigger的名称。第二个是Trigger的组名,第三个是任务開始时间。第四个是结束时间,第五个是反复
 次数(使用SimpleTrigger.REPEAT_INDEFINITELY常量表示无限次),最后一个是反复周期(单位是毫秒),那么这样就创建
 了一个立马并仅仅运行一次的任务。

 scheduler.scheduleJob(jobDetail, trigger);
 这条语句就是把job和Trigger关联,这样当Trigger觉得应该触发的时候就会调用(实际上是Scheduler调用)job.execute方法了。

 scheduler.start();
 千万别忘了加上上面的语句,这条语句通知Quartz使安排的计划生效。

 关于execute方法的參数JobExecutionContext
 JobExecutionContext就和非常多Context结尾的类功能一样。提供的运行时刻的上下文环境。JobExecutionContext中有
 Scheduler,JobDetail,Trigger等非常多对象的引用,从而当你在execute方法内部须须要这些对象的时刻提供的便利。

 JobDetail和Trigger的name和group Scheduler实例相应了非常多job和trigger的实例,为了方便的区分,Quartz使用name和group这两个特性,正如你想向的一样。
 同一个group下不能有两个同样name的JobDetail。Trigger同理
 同一个Scheduler下不能有两个同样group的JobDetail,Trigger同理
 JobDetail和Trigger的全然限定名为:group + name

·  更深入的思考...

HelloWorld的样例还不足以说明一些问题,一些人可能会这样问:假如execute方法中须要一些额外的数据怎么办?比方说execute
 中希望发送一封邮件。可是我须要知道邮件的发送者、接收者等信息?

 存在两种解决方式:
 1.JobDataMap类:
   每一个JobDetail都关联了一个JobDataMap实例,JobDataMap是java.util.Map的子类,基本上是提供key-value形式的数据。并提供了一些便利方法(主要是对java基本数据类型的支持,如put(String key,int value))。当开发者创建JobDetail的时候。能够把附加信息放到JobDataMap中。那么在execute方法中能够依据key找到须要的值。

   JobDetail job = new JobDetail....
   job.getJobDataMap().put("from","snowway@vip.sina.com");
   ...  

在execute中
   String from = jobExecutionContext.getJobDetail().getJobDataMap().getString("from");
   ....

   只是,当你使用数据库存储JobDetail的时候(默认情况下使用RAM),这里有一个致命的弱点。你不能把没有实现java.io.Serializable的对象放入JobDataMap中。由于Quartz将使用Blob字段保存(也能够通过配置文件关闭)序列化过的JobDataMap中的对象。比方你在execute方法中须要一个java.sql.Connection接口实例。这样的情况也是普遍的,那么通常情况下你不能把Connection放入JobDataMap。即使你仅仅想在execute中使用。

(注:读者可临时觉得上面这段话是正确的。然而能够通过指示quartz改变这样的行为。那属于高级话题)

 2.假如你须要一个java.sql.Connection,用于在execute中完毕某些操作。那么你能够把Connection放入Quartz的SchedulerContext中,execute也能够訪问,而且Quartz不会持久化SchedulerContext中的不论什么东西。

   scheduler.getContext().put("java.sql.Connection",connection);  

 execute中
   Connection con = (Connection)jobExecutionContext.getScheduler().getContext().get("java.sql.Connection");

Java 中已经有一个 timer 类能够用来进行运行计划。定时任务。我们所要做的仅仅是 继承 java.util.TimerTask 类。例如以下所看到的:

package com.yourcompany.scheduling; 

 

import java.util.Calendar; 

import java.util.Date;

import java.util.Timer;

import java.util.TimerTask;

 

public class ReportGenerator extends TimerTask { 

 

 public void run() { 

    System.out.println("Generating report");

    //TODO generate report

 }

 

 

class MainApplication { 

 

 public static void main(String[] args) {

    Timer timer new Timer();

    Calendar date = Calendar.getInstance();

    date.set(

      Calendar.DAY_OF_WEEK,

      Calendar.SUNDAY

    );

    date.set(Calendar.HOUR, 0);

    date.set(Calendar.MINUTE, 0);

    date.set(Calendar.SECOND, 0);

    date.set(Calendar.MILLISECOND, 0);

    // Schedule to run every Sunday in midnight

    timer.schedule(

      new ReportGenerator(),   // TimerTask

      date.getTime(),          // Timer

      1000 * 60 * 60 * 24 * 7   // delay

    );

 }

}

 

 

 这里有几个问题。我们的类继承了TimerTask ,而timerTask 也是实现了 java.lang.Runnable 接口。

我们所要做的仅仅是在我们自己的类里重置 run() 方法。所以我们的TimerTask类事实上是一种线程,但线程的调度往往不是依照我们希望来实现的,由于一些垃圾收集等原因,我们计划的时间点。却没有运行必要的任务。

这样会产生一些问题。尽管,Timer 类也提供了scheduleAtFixedRate() 方法用来在垃圾收集后能够高速的追上任务进度,但这个不一定是我们所须要的。特别是在 一些 J2EE server上 Timer 是无法控制的,由于它不在容器的权责范围内。

另外的。这个任务调度也缺乏一些企业级所须要的 特殊 日期定制的功能,以及改动,查找任务的功能。

       这里我们要介绍的是一个开源项目:Quartz 。

       Quartz 定义了两种 基本接口 Job 和 Trigger 。 看名字也就知道,我们的任务必须实现 Job, 我们的时间触发器定义在 Trigger 内。

 看一个样例或许能更快的了解他的用法: package com.yourcompany.scheduling;

 

 

import org.quartz.*; 

 

public class QuartzReport implements Job { 

 

 public void execute(JobExecutionContext cntxt) //必须实现的方法 

    throws JobExecutionException {

      System.out.println(

        "Generating report - " +

cntxt.getJobDetail().getJobDataMap().get("type")

      );

      //TODO Generate report

 }

 

 public static void main(String[] args) { 

    try {

      SchedulerFactory schedFact 

       new org.quartz.impl.StdSchedulerFactory();

      Scheduler sched = schedFact.getScheduler();

      sched.start();

      JobDetail jobDetail =

        new JobDetail(

          "Income Report",       // 任务名

          "Report Generation", // 任务组

          QuartzReport.class    //任务运行的类

        );

      jobDetail.getJobDataMap().put(

                                "type",   

                                "FULL"

                               );

 

      CronTrigger trigger new CronTrigger( 

        "Income Report",              //触发器名

        "Report Generation"          //触发器组

      );

      trigger.setCronExpression(     // 触发器时间设定

        "0 0 12 ? * SUN"

      );

 

      sched.scheduleJob(jobDetail, trigger); // 运行任务 

 

    } catch (Exception e) {

      e.printStackTrace();

    }

 }

}

这里面我们能够看到。当我们定义了任务运行 QuartzReport 类后。须要定一个Scheduler类用来运行计划任务。

一个JobDetail 类来描写叙述这个任务的信息,包含任务信息,任务所在组。任务运行的类。

然后还要定义一个 触发器。类似的也包含触发器名,触发器所在组,触发器触发时间设定。

最后是调度器Scheduler类运行计划任务。

基本上一个计划任务运行的流程就完毕了。

当然。我们还看到了上面红色代表的内容。这些内容主要是提供在job方法运行的时候所须要的參数的提供。这里使用了JobDataMap 类,它事实上就是实现了map的特殊应用的一个类。用法与Map 非常类似。我们能够用 put() 输入參数。在Job类中使用cntxt.getJobDetail().getJobDataMap().get("type") 方法获取输入的參数的值。这里的cntxt 是JobExecutionContext 。

就是包含任务运行上下文的一个信息类。这样我们的一个主要的任务运行就能够搞定了。

 

点  。CronTrigger的时间设置说明在最后来介绍。

 

以下我们介绍一下在 J2EE 环境下怎样来使用 Quartz 。

首先。我们要配置 web.xml ,加入 一下内容。主要是Quartz 的初始化,

<servlet>

          <servlet-name>QuartzInitializer</servlet-name>

          <display-name>Quartz Initializer Servlet</display-name>

          <servlet-class>org.quartz.ee.servlet.QuartzInitializerServlet</servlet-class>

          <load-on-startup>1</load-on-startup>

     </servlet>

 

然后还要有一个Quartz 的配置文件 quartz.properties 放置在 WEB-INF/classes文件夹以下。

StdScheduleFactory()会读取它。配置例如以下 #

 

# Configure Main Scheduler Properties 

#

 

org.quartz.scheduler.instanceName = TestScheduler 

org.quartz.scheduler.instanceId = one

 

# Configure ThreadPool 

#

 

org.quartz.threadPool.class = org.quartz.simpl.SimpleThreadPool 

org.quartz.threadPool.threadCount = 5

org.quartz.threadPool.threadPriority = 4

 

# Configure JobStore 

#

 

org.quartz.jobStore.misfireThreshold = 5000

org.quartz.jobStore.class = org.quartz.simpl.RAMJobStore

 

这里我们使用的 RAMJobStore 存储方式。这样假设我们的webserver重新启动的话,我们全部未运行的任务信息都回丢失。当然,我们也有另外的选择,我们能够把这样的信息存储在数据库内,就是使用JDBCJobStoreTX

#

# Configure ThreadPool 

#

org.quartz.jobStore.class = org.quartz.impl.jdbcjobstore.JobStoreTX

 

org.quartz.jobStore.driverDelegateClass = org.quartz.impl.jdbcjobstore.PostgreSQLDelegate

org.quartz.jobStore.dataSource = myDS

org.quartz.jobStore.tablePrefix = QRTZ_

 

# Configure Datasources 

#

 

org.quartz.dataSource.myDS.driver = org.postgresql.Driver

org.quartz.dataSource.myDS.URL = jdbc:postgresql:dev

org.quartz.dataSource.myDS.user = dejanb

org.quartz.dataSource.myDS.password =

org.quartz.dataSource.myDS.maxConnections 5

 

附:cronExpression配置说明

字段

 

同意值

 

同意的特殊字符

 

0-59

 

, - * /

 

0-59

 

, - * /

小时

 

0-23

 

, - * /

日期

 

1-31

 

, - * ? / L W C

月份

 

1-12 或者 JAN-DEC

 

, - * /

星期

 

1-7 或者 SUN-SAT

 

, - * ? / L C #

年(可选)

 

留空, 1970-2099

 

, - * /

 

Cron 的小小说明

表示方式

意义

"0 0 12 * * ?"

Fire at 12pm (noon) every day

"0 15 10 ?

 * *"

Fire at 10:15am every day

"0 15 10 * * ?

"

Fire at 10:15am every day

"0 15 10 * * ?

 *"

Fire at 10:15am every day

"0 15 10 * * ? 2005"

Fire at 10:15am every day during the year 2005

"0 * 14 * * ?"

Fire every minute starting at 2pm and ending at 2:59pm, every day

"0 0/5 14 * * ?"

Fire every 5 minutes starting at 2pm and ending at 2:55pm, every day

"0 0/5 14,18 * * ?"

Fire every 5 minutes starting at 2pm and ending at 2:55pm, AND fire every 5 minutes starting at 6pm and ending at 6:55pm, every day

"0 0-5 14 * * ?"

Fire every minute starting at 2pm and ending at 2:05pm, every day

"0 10,44 14 ? 3 WED"

Fire at 2:10pm and at 2:44pm every Wednesday in the month of March.

"0 15 10 ? * MON-FRI"

Fire at 10:15am every Monday, Tuesday, Wednesday, Thursday and Friday

"0 15 10 15 * ?

"

Fire at 10:15am on the 15th day of every month

"0 15 10 L * ?

"

Fire at 10:15am on the last day of every month

"0 15 10 ? * 6L"

Fire at 10:15am on the last Friday of every month

"0 15 10 ?

 * 6L"

Fire at 10:15am on the last Friday of every month

"0 15 10 ?

 * 6L 2002-2005"

Fire at 10:15am on every last friday of every month during the years 2002, 2003, 2004 and 2005

"0 15 10 ?

 * 6#3"

Fire at 10:15am on the third Friday of every month

 

 

 

 

 

 

 

 

Quartz学习笔记的更多相关文章

  1. Quartz学习笔记:集群部署&高可用

    Quartz学习笔记:集群部署&高可用 集群部署 一个Quartz集群中的每个节点是一个独立的Quartz应用,它又管理着其他的节点.这就意味着你必须对每个节点分别启动或停止.Quartz集群 ...

  2. Quartz学习笔记:基础知识

    Quartz学习笔记:基础知识 引入Quartz 关于任务调度 关于任务调度,Java.util.Timer是最简单的一种实现任务调度的方法,简单的使用如下: import java.util.Tim ...

  3. 作业调度框架 Quartz 学习笔记(三) -- Cron表达式 (转载)

    前面两篇说的是简单的触发器(SimpleTrigger) , SimpleTrigger 只能处理简单的事件出发,如果想灵活的进行任务的触发,就要请出 CronTrigger 这个重要人物了. Cro ...

  4. quartz学习笔记(一)简单入门

    前言 quartz是Java编写的一款开源的任务调度开发框架,在项目开发中很多场景都可以用到,比如订单超期自动收货. 所谓程序源于生活,生活中也有很多场景可以用quartz来模拟,比如工作日早上七点起 ...

  5. Quartz学习笔记1:Quartz概述

    Quartz是开源任务调度框架中的翘楚,它提供了强大的 任务调度机制.Quartz允许开发人员灵活的定义触发器的调度时间表,并可对触发器和任务进行关联映射.此外,Quartz提供了调度运行环境的持久化 ...

  6. 【转】Spring.NET学习笔记——目录

    目录 前言 Spring.NET学习笔记——前言 第一阶段:控制反转与依赖注入IoC&DI Spring.NET学习笔记1——控制反转(基础篇) Level 200 Spring.NET学习笔 ...

  7. Java学习笔记 -- Java定时调度工具Timer类

    1 关于 (时间宝贵的小姐姐请跳过) 本教程是基于Java定时任务调度工具详解之Timer篇的学习笔记. 什么是定时任务调度 基于给定的时间点,给定的时间间隔或者给定的执行次数自动执行的任务. 在Ja ...

  8. Spring.NET学习笔记——目录(原)

    目录 前言 Spring.NET学习笔记——前言 第一阶段:控制反转与依赖注入IoC&DI Spring.NET学习笔记1——控制反转(基础篇) Level 200 Spring.NET学习笔 ...

  9. 我的Android进阶之旅------>Android中编解码学习笔记

    编解码学习笔记(一):基本概念 媒体业务是网络的主要业务之间.尤其移动互联网业务的兴起,在运营商和应用开发商中,媒体业务份量极重,其中媒体的编解码服务涉及需求分析.应用开发.释放license收费等等 ...

随机推荐

  1. HTML如何创建二级目录

    #classify ul li div{width:100px;  height:200px;  display:none;  position:absolute;  left:100px;  top ...

  2. angular4 动态Form中获取表单字段并在页面中使用的方法

    主要有两种方式 第一种 使用get属性 页面中使用如下: 第二种 使用普通方法事件  页面中使用如下 *转载请附出处

  3. 11.6八校联考T1,T2题解

    因为版权问题,不丢题面,不放代码了(出题人姓名也隐藏) T1 这,是一道,DP题,但是我最开始看的时候,我思路挂了,以为是一道简单题,然后就写错了 后来,我正确理解题意后写了个dfs,幸亏没有记忆化, ...

  4. 【JAVAWEB学习笔记】30_WEB总结_思维导图

    可以在浏览器放大来查看细节,或者另存为图片到本地电脑查看.

  5. 【SpringBoot】关闭HttpClient无用日志

    环境: SpringBoot pom依赖了apache.commons.HttpClient: <!--httpclient--> <dependency> <group ...

  6. 【BZOJ 4031】 4031: [HEOI2015]小Z的房间 (Matrix-Tree Theorem)

    4031: [HEOI2015]小Z的房间 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 1089  Solved: 533 Description ...

  7. luoguP4284 [SHOI2014]概率充电器 概率期望树形DP

    这是一道告诉我概率没有想象中那么难的题..... 首先,用期望的线性性质,那么答案为所有点有电的概率和 发现一个点的有电的概率来源形成了一个"或"关系,在概率中,这并不好计算... ...

  8. POJ2222 Keywords Search AC自动机模板

    http://acm.hdu.edu.cn/showproblem.php?pid=2222 题意:给出一些单词,求多少个单词在字符串中出现过(单词表单词可能有相同的,这些相同的单词视为不同的分别计数 ...

  9. Codeforces Round #245 (Div. 2) A. Points and Segments (easy) 贪心

    A. Points and Segments (easy) Time Limit: 1 Sec  Memory Limit: 256 MB 题目连接 http://codeforces.com/con ...

  10. 如何解决…has been modified since the precompiled header… was built的问题

    如何解决…has been modified since the precompiled header… was built 的问题 xcode5.1在程序中报错: File '/Applicatio ...