目录
三、Quartz
一、Java定时任务介绍
在Java中,用得比较多的有两种,一个是Timer,一个是Quartz;
其中Timer是这是jdk自带的类库,一般用来实现简单的定时调度,由一个后台线程进行任务的调度,所以对于并发调度不友好;
Quartz不是jdk自带的,但是他的功能更加强大,一般用于比较复杂的定时调度,可以解决Timer的并发调度问题;
二、Timer
2.1、Timer与TimerTask
Timer是在jdk自带的工具类,Timer为与java.util包下,可以将其理解为一个“定时器”;
当定时器到点后,执行的任务是TimerTask,称为“任务”;
下面就是一个TimerTask示例,注意需要重写TimerTask的run方法,类似于Runnable接口的run方法。
package cn.ganlixin.task;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.TimerTask;
public class MyTask extends TimerTask {
@Override
public void run() {
SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
System.out.println(dateFormat.format(new Date()));
}
}
下面是一个简单的定时执行任务的示例:
package cn.ganlixin.timer;
import cn.ganlixin.task.MyTask;
import org.junit.Test;
import java.text.ParseException;
import java.util.Timer;
public class TestTimer {
@Test
public void testTimerDemo() throws InterruptedException {
// 创建一个定时器
Timer timer = new Timer();
// 5秒之后执行任务,只执行一次,执行MyTask的run方法
// schedule(TimerTask task, long delay)
timer.schedule(new MyTask(), 5 * 1000L);
// 让当前线程阻塞,等待上面的任务执行
Thread.sleep(10000);
}
}
Timer有几个可以设置任务执行的方法:
// 在deley毫秒毫秒后,开始执行task,只执行一次
void schedule(TimerTask task, long delay)
// 在deley毫秒毫秒后,开始执行task,之后每隔period毫秒执行一次task
void schedule(TimerTask task, long delay, long period)
// 在date时刻,开始执行task,只执行一次
void schedule(TimerTask task, Date date)
// 在date时刻,开始执行task,之后每隔period毫秒执行一次task
void schedule(TimerTask task, Date firstTime, long period)
// 和schedule(TimerTask task, long delay, long period)一样,但是以固定速率执行
scheduleAtFixedRate(TimerTask task, long delay, long period)
// 和schedule(TimerTask task, Date firstTime, long period)一样,但是以固定速率执行
scheduleAtFixedRate(TimerTask task, Date firstTime, long period)
schedule和scheduleAtFixRate有一些区别。
2.2、schedule
1、当首次计划执行时间早于当前时间,比如要求在1980年1月1日凌晨执行某个任务
对于这种情况,启动定时任务后,会立即执行任务。下面是例子:
package cn.ganlixin.timer;
import org.junit.Test;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;
public class TestTimer {
@Test
public void testStartBefore() throws InterruptedException, ParseException {
Timer timer = new Timer();
SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
System.out.println("当前时间:" + dateFormat.format(new Date()));
Date date = dateFormat.parse("2019-01-01 01:00:00"); // 开始时间为过去
timer.schedule(new TimerTask() {
@Override
public void run() {
SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
Date scheduledExecuteTime = new Date(scheduledExecutionTime());
System.out.println("预计执行任务时间:" + dateFormat.format(scheduledExecuteTime));
System.out.println("开始任务:" + dateFormat.format(new Date()));
try {
Thread.sleep(10000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("完成任务:" + dateFormat.format(new Date()));
}
}, date);
// 让当前线程阻塞,等待上面的任务执行
Thread.sleep(100000);
}
}
运行输出:
当前时间:2019-03-07 07:56:47
预计执行任务时间:2019-01-01 01:00:00
开始任务:2019-03-07 07:56:47
完成任务:2019-03-07 07:56:57
2、首次执行时间在当前时间之后,但是执行时间超过了period(周期)
比如,有一个任务在4秒后启动,每3秒执行1次,任务每次执行要10秒,此时,因为执行任务所需时间超过了周期时间,所以一次任务执行完后,立即执行下一次任务;
package cn.ganlixin.timer;
import org.junit.Test;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;
public class TestTimer {
@Test
public void testTimerDemo() throws InterruptedException, ParseException {
// 创建一个定时器
Timer timer = new Timer();
SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
System.out.println("当前时间:" + dateFormat.format(new Date()));
timer.schedule(new TimerTask() {
@Override
public void run() {
SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
Date scheduledExecuteTime = new Date(scheduledExecutionTime());
System.out.println("预计执行任务时间:" + dateFormat.format(scheduledExecuteTime));
System.out.println("开始任务:" + dateFormat.format(new Date()));
try {
Thread.sleep(10000); // 休眠10秒,模拟任务执行耗时
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("完成任务:" + dateFormat.format(new Date()));
System.out.println("-----------------------------------------");
}
}, 4 * 1000L, 3 * 1000L);
// 让当前线程阻塞,等待上面的任务执行
Thread.sleep(100000);
}
}
运行结果:
当前时间:2019-03-07 08:14:40
预计执行任务时间:2019-03-07 08:14:44
开始任务:2019-03-07 08:14:44
完成任务:2019-03-07 08:14:54
-----------------------------------------
预计执行任务时间:2019-03-07 08:14:54
开始任务:2019-03-07 08:14:54
完成任务:2019-03-07 08:15:04
-----------------------------------------
预计执行任务时间:2019-03-07 08:15:04
开始任务:2019-03-07 08:15:04
完成任务:2019-03-07 08:15:14
-----------------------------------------
预计执行任务时间:2019-03-07 08:15:14
开始任务:2019-03-07 08:15:14
.........
2.3、scheduleAtFixRate
1、当首次计划执行时间早于当前时间,那么scheduleAtFixRate会将错过的这段时间中,本应该执行的任务“补”回来。
package cn.ganlixin.timer;
import org.junit.Test;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;
public class TestTimer {
@Test
public void testTimerDemo() throws InterruptedException, ParseException {
// 创建一个定时器
Timer timer = new Timer();
SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
System.out.println("当前时间:" + dateFormat.format(new Date()));
Date date = dateFormat.parse("2019-01-01 01:00:00"); // 开始时间为过去
timer.scheduleAtFixedRate(new TimerTask() {
@Override
public void run() {
SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
Date scheduledExecuteTime = new Date(scheduledExecutionTime());
System.out.println("预计执行任务时间:" + dateFormat.format(scheduledExecuteTime));
System.out.println("开始任务:" + dateFormat.format(new Date()));
try {
Thread.sleep(10000); // 休眠10秒,模拟任务执行耗时
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("完成任务:" + dateFormat.format(new Date()));
System.out.println("-----------------------------------------");
}
}, date, 3 * 1000L);
// 让当前线程阻塞,等待上面的任务执行
Thread.sleep(100000);
}
}
2、首次执行时间在当前时间之后,但是执行时间超过了period(周期),此时会将错过的任务补回来
package cn.ganlixin.timer;
import org.junit.Test;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;
public class TestTimer {
@Test
public void testTimerDemo() throws InterruptedException, ParseException {
// 创建一个定时器
Timer timer = new Timer();
SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
System.out.println("当前时间:" + dateFormat.format(new Date()));
timer.scheduleAtFixedRate(new TimerTask() {
@Override
public void run() {
SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
Date scheduledExecuteTime = new Date(scheduledExecutionTime());
System.out.println("预计执行任务时间:" + dateFormat.format(scheduledExecuteTime));
System.out.println("开始任务:" + dateFormat.format(new Date()));
try {
Thread.sleep(10000); // 休眠10秒,模拟任务执行耗时
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("完成任务:" + dateFormat.format(new Date()));
System.out.println("-----------------------------------------");
}
}, 4 * 1000L, 3 * 1000L);
// 让当前线程阻塞,等待上面的任务执行
Thread.sleep(100000);
}
}
运行输出
当前时间:2019-03-07 08:26:53
预计执行任务时间:2019-03-07 08:26:57
开始任务:2019-03-07 08:26:57
完成任务:2019-03-07 08:27:07
-----------------------------------------
预计执行任务时间:2019-03-07 08:27:00
开始任务:2019-03-07 08:27:07
完成任务:2019-03-07 08:27:17
-----------------------------------------
预计执行任务时间:2019-03-07 08:27:03
开始任务:2019-03-07 08:27:17
完成任务:2019-03-07 08:27:27
-----------------------------------------
预计执行任务时间:2019-03-07 08:27:06
开始任务:2019-03-07 08:27:27
完成任务:2019-03-07 08:27:37
三、Quartz