原文作者:一叶丿清风
schedule方法和scheduleAtFixedRate方法都可以实现任务的延时和不延时执行且都会按顺序执行,因此不需要考虑非线程安全的情况。它们的主要区别只在于有没有追赶特性。具体如下:
- 任务执行未超时, 下次执行时间=上次执行开始时间+period;
- 任务执行超时, 下次执行时间=上次执行结束时间;
举个例子:暑假到了老师给schedule和scheduleAtFixedRate两个同学布置作业。老师要求学生暑假每天写2页,30天后完成作业。这两个学生每天按时完成作业,直到第10天,出了意外,两个学生出去旅游花了5天时间,这5天时间里两个人都没有做作业。任务被拖延了。这时候两个学生采取的策略就不同了:schedule重新安排了任务时间,旅游回来的第一天做第11天的任务,第二天做第12天的任务,最后完成任务花了35天。scheduleAtFixedRate是个守时的学生,她总想按时完成老师的任务,于是在旅游回来的第一天把之前5天欠下的任务以及第16天当天的任务全部完成了,之后还是按照老师的原安排完成作业,最后完成任务花了30天。(不一定是第一天就完成之前5天的任务,因任务执行时间而定,但会连续执行任务,不间隔,直到赶上之前时间节点的任务安排为止.)
下面以实示例演示什么是追赶特性。
import java.util.Calendar;
import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;
public class Test {
public static void main(String[] args) throws InterruptedException {
MyTask myTask = new MyTask();
System.out.println("现在执行时间为:" + new Date());
Calendar c = Calendar.getInstance();
c.set(Calendar.SECOND, c.get(Calendar.SECOND) - 20);
Date runDate = c.getTime();
System.out.println("计划执行时间为:" + runDate);
Timer timer = new Timer();
//调用的是schedule方法,验证其不具追赶性
timer.schedule(myTask, runDate, 4000);
}
}
class MyTask extends TimerTask {
@Override
public void run() {
System.out.println("begin timer=" + new Date());
System.out.println("end timer=" + new Date());
}
}
部分运行结果如图 5-1所示
时间“Fri Aug 11 19:43:47 CST 2017”到“Fri Aug 11 19:44:07 CST 2017”之间的时间所对应的Task任务就被取消掉,不被执行了,这就是Task任务不追赶。
验证scheduleAtFixedRate方法具有追赶执行性:
import java.util.Calendar;
import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;
public class Test {
public static void main(String[] args) throws InterruptedException {
MyTask myTask = new MyTask();
System.out.println("现在执行时间为:" + new Date());
Calendar c = Calendar.getInstance();
c.set(Calendar.SECOND, c.get(Calendar.SECOND) - 20);
Date runDate = c.getTime();
System.out.println("计划执行时间为:" + runDate);
Timer timer = new Timer();
//调用scheduleAtFixedRate方法,测试其具有追赶性
timer.scheduleAtFixedRate(myTask, runDate, 4000);
}
}
class MyTask extends TimerTask {
@Override
public void run() {
System.out.println("begin timer=" + new Date());
System.out.println("end timer=" + new Date());
}
}
部分运行结果如图 5-2 所示:
将两个时间段内所对应的Task任务被“补充性”地执行,这就是Task任务追赶特性。