Java并发编程—schedule方法和scheduleAtFixedRate方法的区别

原文作者:一叶丿清风

原文地址:schedule方法和scheduleAtFixedRate方法的区别

schedule方法和scheduleAtFixedRate方法都可以实现任务的延时和不延时执行且都会按顺序执行,因此不需要考虑非线程安全的情况。它们的主要区别只在于有没有追赶特性。具体如下:

  1. 任务执行未超时, 下次执行时间=上次执行开始时间+period;
  2. 任务执行超时, 下次执行时间=上次执行结束时间;

举个例子:暑假到了老师给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所示 

这里写图片描述
图 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 所示: 

这里写图片描述
图 5-2 追赶

将两个时间段内所对应的Task任务被“补充性”地执行,这就是Task任务追赶特性。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值