一、前言

在之前的文章《Java定时任务调度(1)TimerTask原理与实战》中已经介绍了一种实现定时任务调度的方法——Java原生提供的 TimerTask,这个工具适用于一些简单的业务需求。

回顾一下,TimerTask的原理总结起来就是下面这个图:

java计划任务框架 java任务编排框架_java

其实仔细想想,是不是可以抽象出来几个部分?

任务调度,首先要有任务,TimerTask就是 具体任务。有了任务之后是不是要给这个任务设置一下啥时候调,多久调一次,啥时候终止啊?这一步可以叫做 条件设置。那么剩下的就是具体执行,可以叫做 执行器

java计划任务框架 java任务编排框架_java计划任务框架_02

其实就具体代码上来说,后两个都是调Timer的方法

Timer timer = new Timer();
    long delay = 0;
    long period = 1000;
    timer.schedule(timerTask, delay, period);
复制代码

但是,Timer存在一个严重的问题,它是 单线程

如果你有多个任务,且第一个任务执行时间超过了两个任务的间隔时间,那么第二个任务就不能按时执行,此时可以选择使用ScheduledExecutorService来解决,它的内部是个线程池,此外它也能解决上篇文章中提到的Timer异常退出问题。

二、Quartz的简单介绍

那么接下来,就按照这个思路来介绍一个更强大的Java任务调度框架——Quartz

Quartz有一个官方版的简单介绍:

Quartz 允许开发人员根据时间间隔(或天)来调度作业。它实现了作业和触发器的多对多关系,还能把多个作业与不同的触发器关联。整合了 Quartz 的应用程序可以重用来自不同事件的作业,还可以为一个事件组合多个作业。

 如果从来没有接触过任务调度的东西,光看这肯定也懵,不过相信你看完全篇再回过头来,就明白了。

说来也巧,Quartz的设计思路就是上面总结的三部分,只不过名字不同。

  • Job:具体要执行的任务逻辑
  • Trigger:触发器,设置Job的触发条件、间隔,终止时间
  • Scheduler:调度器,启动Trigger执行Job

这么看是不是和前言中对TimerTask的总结一致了?

Quartz Demo的创建

如果使用传统方式,就是去官方下载一个jar包,新建一个Java project,导包就可以了。 现在一般都是使用自动化构建工具了。

打开start.spring.io/ 填写相关配置,这里我使用Maven

java计划任务框架 java任务编排框架_java任务调度框架_03

当然此处也可以选择右侧的add dependencies,那我一般都是自己在pom文件中添加依赖。

导入依赖

<dependency>
	    <groupId>org.quartz-scheduler</groupId>
	    <artifactId>quartz</artifactId>
	    <version>2.3.2</version>
	</dependency>
复制代码

写Demo

public static void main(String[] args) {
		 try {
             // 工厂方法获取默认的调度器
             Scheduler scheduler = StdSchedulerFactory.getDefaultScheduler();

             // 启动
             scheduler.start();
             //关闭
             scheduler.shutdown();

         } catch (SchedulerException se) {
             se.printStackTrace();
         }
	}
复制代码

其实这个例子是官网提供的【www.quartz-scheduler.org/documentati… 很简单,一共就三行代码。

执行结果是什么那?

22:47:27.348 [main] INFO org.quartz.impl.StdSchedulerFactory - Using default implementation for ThreadExecutor
22:47:27.355 [main] INFO org.quartz.simpl.SimpleThreadPool - Job execution threads will use class loader of thread: main
22:47:27.370 [main] INFO org.quartz.core.SchedulerSignalerImpl - Initialized Scheduler Signaller of type: class org.quartz.core.SchedulerSignalerImpl
22:47:27.370 [main] INFO org.quartz.core.QuartzScheduler - Quartz Scheduler v.2.3.2 created.
22:47:27.371 [main] INFO org.quartz.simpl.RAMJobStore - RAMJobStore initialized.
22:47:27.372 [main] INFO org.quartz.core.QuartzScheduler - Scheduler meta-data: Quartz Scheduler (v2.3.2) 'DefaultQuartzScheduler' with instanceId 'NON_CLUSTERED'
  Scheduler class: 'org.quartz.core.QuartzScheduler' - running locally.
  NOT STARTED.
  Currently in standby mode.
  Number of jobs executed: 0
  Using thread pool 'org.quartz.simpl.SimpleThreadPool' - with 10 threads.
  Using job-store 'org.quartz.simpl.RAMJobStore' - which does not support persistence. and is not clustered.

22:47:27.372 [main] INFO org.quartz.impl.StdSchedulerFactory - Quartz scheduler 'DefaultQuartzScheduler' initialized from default resource file in Quartz package: 'quartz.properties'
22:47:27.372 [main] INFO org.quartz.impl.StdSchedulerFactory - Quartz scheduler version: 2.3.2
22:47:27.372 [main] INFO org.quartz.core.QuartzScheduler - Scheduler DefaultQuartzScheduler_$_NON_CLUSTERED started.
22:47:27.372 [main] INFO org.quartz.core.QuartzScheduler - Scheduler DefaultQuartzScheduler_$_NON_CLUSTERED shutting down.
22:47:27.372 [main] INFO org.quartz.core.QuartzScheduler - Scheduler DefaultQuartzScheduler_$_NON_CLUSTERED paused.
22:47:27.372 [main] DEBUG org.quartz.simpl.SimpleThreadPool - Shutting down threadpool...
22:47:27.372 [DefaultQuartzScheduler_QuartzSchedulerThread] DEBUG org.quartz.core.QuartzSchedulerThread - batch acquisition of 0 triggers
22:47:27.372 [main] DEBUG org.quartz.simpl.SimpleThreadPool - Shutdown of threadpool complete.
22:47:27.372 [main] INFO org.quartz.core.QuartzScheduler - Scheduler DefaultQuartzScheduler_$_NON_CLUSTERED shutdown complete.
22:47:27.868 [DefaultQuartzScheduler_Worker-6] DEBUG org.quartz.simpl.SimpleThreadPool - WorkerThread is shut down.
22:47:27.868 [DefaultQuartzScheduler_Worker-10] DEBUG org.quartz.simpl.SimpleThreadPool - WorkerThread is shut down.
22:47:27.868 [DefaultQuartzScheduler_Worker-5] DEBUG org.quartz.simpl.SimpleThreadPool - WorkerThread is shut down.
22:47:27.868 [DefaultQuartzScheduler_Worker-4] DEBUG org.quartz.simpl.SimpleThreadPool - WorkerThread is shut down.
22:47:27.868 [DefaultQuartzScheduler_Worker-3] DEBUG org.quartz.simpl.SimpleThreadPool - WorkerThread is shut down.
22:47:27.869 [DefaultQuartzScheduler_Worker-2] DEBUG org.quartz.simpl.SimpleThreadPool - WorkerThread is shut down.
22:47:27.869 [DefaultQuartzScheduler_Worker-1] DEBUG org.quartz.simpl.SimpleThreadPool - WorkerThread is shut down.
22:47:27.868 [DefaultQuartzScheduler_Worker-7] DEBUG org.quartz.simpl.SimpleThreadPool - WorkerThread is shut down.
22:47:27.868 [DefaultQuartzScheduler_Worker-8] DEBUG org.quartz.simpl.SimpleThreadPool - WorkerThread is shut down.
22:47:27.868 [DefaultQuartzScheduler_Worker-9] DEBUG org.quartz.simpl.SimpleThreadPool - WorkerThread is shut down.

复制代码

输出的结果还是有很多看头的

Quartz Scheduler v.2.3.2 created  可以看到使用版本
RAMJobStore initialized  可以看到存储方式是RAMJobStore
最后面的SimpleThreadPool 就是线程池,一共有1-10   10个线程
复制代码

看到线程池,你会想到什么?

可以参考我的上篇文章:《Java定时任务调度(1)TimerTask原理与实战》

假如我把第三行代码注释掉,会发生什么? 注意箭头的位置,这个程序并没有停止。

java计划任务框架 java任务编排框架_quartz任务调度框架_04

这里官网也有解释:

Once you obtain a scheduler using StdSchedulerFactory.getDefaultScheduler(), your application will not terminate until you call scheduler.shutdown(), because there will be active threads.

一旦使用StdSchedulerFactory.getDefaultScheduler()获得调度程序,在调用scheduler.shutdown()之前,应用程序不会终止,因为会有活动线程。

有了启动和停止,任务的具体执行自然就在这中间来完成。

java计划任务框架 java任务编排框架_quartz任务调度框架_05

把官网的例子复制过来,很明显,出现了错误。 具体原因就是一个是没有HelloJob这个类,另一个静态导入要补上

public class HelloJob implements Job{

	@Override
	public void execute(JobExecutionContext context) throws JobExecutionException {
		// TODO Auto-generated method stub
		
		SimpleDateFormat  simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
		
		System.out.println("你好!当前时间是:"+simpleDateFormat.format(new Date()));
	}
	
}


//注意,静态导入不在这个HelloJob类中

import static org.quartz.JobBuilder.*;
import static org.quartz.TriggerBuilder.*;
import static org.quartz.SimpleScheduleBuilder.*;
复制代码

当然在执行之前,一定要在停止的前面加上Thread.sleep(),要不然,直接就关闭了。

为了尽快的看到效果,我把间隔时间设置为5秒

效果:

java计划任务框架 java任务编排框架_java任务调度框架_06

好了,简单的Demo就写好了。