初识
官方文档对Quartz的描述:
Quartz是一个拥有丰富特性,开源的作业调度类库。小到独立的java应用程序,大到电子商务系统都可以将Quartz整合到其中。Quartz可以创建简单或者复杂的调度程序来执行作业,而作业作为java的组件来执行你希望实现的任何功能。
我的描述:Quartz就是在对的时间做对的事。
再看
在Quartz官网下载了他的文档,有好几篇,看着也真是够够的了。其实我认为只需要了解三个核心的内容,就可以对Quartz进行使用了。三个核心内容是什么呢?作业,触发器,调度程序。下面我们依次聊聊这三项。
作业(Job)
作业就是你希望在对的时间做什么事。在Quartz中作业的定义很简单,只需要实现Job接口,并实现它的execute方法就可以定义你希望做的事情。
例如:
public class HelloJob implements Job {
Logger logger=LoggerFactory.getLogger(HelloJob.class);
@Override
public void execute(JobExecutionContext arg0) throws JobExecutionException {
logger.info("Hello World:"+new Date());
}
}
关于作业的存储:在Quartz中对作业有多种的持久化操作。
- JobStore接口,它提供了多种多样的机制来存储作业。
- JDBCJobStore接口,它可以将作业通过JDBC存储到数据库中。
- RAMJobStore,通常我们使用的是RAMJobStore,将作业以及触发器存储在内存中。
触发器(Trigger)
触发器会在你希望的对的时间对作业进行触发。在Quartz中,已经定义好了一系列的builder类,它们可以被开发人员用来创建相应的触发器,作业,作业调度等对象。在创建触发器的时候,我们只需要使用TriggerBuilder即可创建Trigger即可。在实际应用中,比较常用的两种触发器是SimpleTrigger和CronTrigger。
- SimpleTrigger:用于简单的触发以及重复执行作业任务。首先看一段代码:
@Test
public void test04(){
logger.info("--------初始化调度程序-----");
Scheduler scheduler=getScheduler();//通过工厂类SchedulerFactory创建Scheduler,详见后文
logger.info("--------初始化调度程序完成-----");
//返回当前时间加10秒
Date triggerStartTime=DateBuilder.nextGivenSecondDate(null, 10);
JobDetail job=JobBuilder.newJob(SimpleJob.class).withIdentity("job", "group").build();
//设定5秒重复触发一次,直到永远
SimpleTrigger simpleTrigger=(SimpleTrigger) TriggerBuilder.newTrigger().
withIdentity("trigger", "group").startAt(triggerStartTime).
withSchedule(SimpleScheduleBuilder.repeatSecondlyForever(5)).
build();
logger.info(job+"--------"+simpleTrigger.getRepeatCount());
try {
scheduler.scheduleJob(job, simpleTrigger);
logger.info("--------作业调度开始---------");
scheduler.start();
//这里sleep的目的是避免主线程在触发器触发之前就结束了
try {
Thread.sleep(100000);
} catch (InterruptedException e) {
e.printStackTrace();
}
logger.info("------- 准备结束 -------------------");
scheduler.shutdown(true);
logger.info("------- 结束完成 -------------------");
} catch (SchedulerException e) {
e.printStackTrace();
}
}
正如上面代码所展示的,使用DateBuilder类创建一个触发时间,使用JobBuilder创建一个简单的作业任务,再使用TriggerBuilder来创建一个简单触发器,这个触发器会在DateBuilder创建的时间开始,根据SimpleScheduleBuilder所设定的每5秒触发一次,直到永远地执行JobBuilder所创建的作业任务。
基本的使用就是这样咯。当然,触发器的创建以及作业调度的方式方法还有很多,具体的使用还请您查阅Quartz的API,在此就不赘述了。
- CronTrigger:使用cron表达式来进行计划作业任务设定的触发器。
刚开始接触cron表达式的时候很是让人抓狂,因为正如Oracle对它的描述一样,在一个cron表达式中会包含6或者7个域。对于第一次接触这种描述方式的表达式,很容易就被到底这六个值或七个值代表了什么意思?而此时我学习的方法是外事问谷哥,内事找度娘。百度了很多cron表达式的例子,见识的多了渐渐的也就了解了cron表达式的含义。关于cron表达式的内容在此也不赘述,我本身也是对其不甚了解,您可以参阅这两篇文章学习,《cron表达式的解释》以及《cron表达式详解》
在Quartz中创建一个CronTrigger与创建SimpleTrigger类似,都是使用TriggerBuilder来进行创建,例如下面这段代码所展示:
@Test
public void test05(){
logger.info("-----初始化调度程序-----");
Scheduler scheduler = getScheduler();
logger.info("-----初始化完成-----");
/*
* cron表达式有两种格式:
* 1.秒 分 时 月中天 月 周中天 年
* 2.秒 分 时 月中天 月 周中天
*/
JobDetail cronJob=JobBuilder.newJob(CronJob.class).withIdentity("cronJob", "group").build();
//cron表达式的意思为15点51分50秒
CronTrigger cronTrigger=(CronTrigger) TriggerBuilder.newTrigger().withIdentity("cronTrigger", "group").
withSchedule(CronScheduleBuilder.cronSchedule("50 51 15 * * ?")).build();
try {
scheduler.scheduleJob(cronJob, cronTrigger);
scheduler.start();
try {
Thread.sleep(100000);
} catch (InterruptedException e) {
e.printStackTrace();
}
logger.info("------- 准备结束 -------------------");
scheduler.shutdown(true);
logger.info("------- 结束完成 -------------------");
} catch (SchedulerException e) {
e.printStackTrace();
}
}
关于CronTrigger没有特别要强调的,只需要学会cron表达式的编写即可。
作业调度(Scheduler)
作业调度就是将触发器与作业任务整合起来,在正确的时间调用触发器去触发执行作业任务。
在Quartz中,通常使用工厂类来获取作业调度对象,如下:
private Scheduler getScheduler() {
SchedulerFactory factory=new StdSchedulerFactory();
Scheduler scheduler=null;
try {
scheduler=factory.getScheduler();
} catch (SchedulerException e) {
e.printStackTrace();
}
return scheduler;
}
如上例所展示,使用的是标准工厂StdSchedulerFactory来获取Scheduler 对象。
Scheduler 对象也有其自己的生命周期,它的生命周期从factory.getScheduler()开始,直到scheduler.shutdown()时结束。一旦调用了shutdown方法,那么线程及其他资源都会释放,那么就无法再获取到这个scheduler对象了。如果希望暂停一下任务的执行,那么可以调用suspend方法来挂起任务执行。
总结
至此,关于Quartz的基本的学习了解就到一段落了。我认为只要了解了上述的关于Quartz的基本的三个元素,那么再根据具体的需求,对应Quartz的API,就可以很好的解决有关计划任务的问题了。
来源:oschina
链接:https://my.oschina.net/u/3101476/blog/1555870