传统定时器技术 Timer and TimerTask

本文为张孝祥java并发课程的学习笔记。
java.util.Timer定时器,实际上是个线程,定时调度所拥有的TimerTasks。
一个TimerTask实际上就是一个拥有run方法的类,需要定时执行的代码放到run方法体内,TimerTask一般是以匿名类的方式创建。

我们看看看Timer的api文档

它拥有以下的几个方法

cancel,取消任务

purge,移除已经取消的任务



schedule(TimerTask task, long delay)

delay毫秒后执行task任务。
再看看TimerTask
public abstract class TimerTask extends Object
implements Runnable
即使是猜,我们应该也能猜到TimerTask的关键就在里面的run方法。
ok我们看一个例子
import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;

public class TraditionalTimerTest2 {

    
    public static void main(String[] args) {
    new Timer().schedule(new TimerTask() {
            @Override
            public void run() {
                System.out.println("bombing!");
            }
        }, 1000);  //1秒后 执行run方法
        
        //主线程继续运行 每隔1秒 输出现在的秒数字
        while(true){
            System.out.println(new Date().getSeconds());
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
        }
    }

}


运行结果
---------- 运行java ----------
29
bombing!
30
31
32
33
34
35
36
37
38
39
40
41
42

schedule(TimerTask task, long delay, long period)

这个是做什么的? delay毫秒后开始执行task任务,然后每隔period毫秒执行一次
部分代码如下:
    public static void main(String[] args) {
    new Timer().schedule(new TimerTask() {
            @Override
            public void run() {
                System.out.println("bombing!");
            }
        }, 3000,1000);
        ....
}
结果
---------- 运行java ----------
5
6
7
bombing!
8
bombing!
9
bombing!
10
bombing!
11
bombing!
当然还有另一种方式来达到上面的效果
    public static void main(String[] args) {

        class MyTimerTask extends TimerTask{
            
            @Override
            public void run() {
                System.out.println("bombing!");
                new Timer().schedule(new MyTimerTask(),1000);
            }
        }
        
        new Timer().schedule(new MyTimerTask(), 2000);
    }
大家应该能猜出结果吧。
如果我让炸弹两秒爆炸一次,然后隔三秒再爆一次,然后两秒,然后三秒...
如何?
public class TraditionalTimerTest4 {

    private static int count = 0;
    public static void main(String[] args) {

        class MyTimerTask extends TimerTask{
            
            @Override
            public void run() {
                count = (count+1)%2;
                System.out.println("bombing!");
                new Timer().schedule(new MyTimerTask(),2000+1000*count);
            }
        }
        new Timer().schedule(new MyTimerTask(), 2000);
        .....
    }
}

运行结果
---------- 运行java ----------
17
18
19
20
bombing
21
22
bombing
23
24
25
bombing
26
27
bombing
28
29
30
bombing
31
32
bombing
33
34
35

当然还有一种方式
import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;
    public class MyTimerTask2 extends TimerTask{
            
        @Override
        public void run() {
            System.out.println("bombing!2");
            new Timer().schedule(new MyTimerTask3(),2000);
        }
    }
import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;

    public class MyTimerTask3 extends TimerTask{
            
        @Override
        public void run() {
            System.out.println("bombing!3");
            new Timer().schedule(new MyTimerTask2(),3000);
        }
    }


有个这两个类大家应该明白怎么写了吧。

schedule(TimerTask task, Date time)

schedule(TimerTask task, Date firstTime, long period)
上面那个方法是time时刻执行task,下面那个是firstTime执行一次,每隔period毫秒再执行一回。

如果我想让炸弹每天早上8:00炸一次。明白怎么写了吧?firsttime指定为明天早上,period为24*86400*1000。ok?


schedule和scheduleAtFixedRate的区别在于,如果指定开始执行的时间在当前系统运行时间之前,scheduleAtFixedRate会把已经过去的时间也作为周期执行,而schedule不会把过去的时间算上。

比如
SimpleDateFormat fTime = new SimpleDateFormat("yyyy/MM/dd HH:mm:ss");
  Date d1 = fTime.parse("2005/12/30 14:10:00");
 
  t.scheduleAtFixedRate(new TimerTask(){
   public void run()
   {
       System.out.println("this is task you do6");
   }
  },d1,3*60*1000);

间隔时间是3分钟,指定开始时间是2005/12/30 14:10:00,如果我在14:17:00分执行这个程序,那么会立刻打印3次

this is task you do6     //14:10
this is task you do6     //14:13
this is task you do6     //14:16

并且注意,下一次执行是在14:19 而不是 14:20。就是说是从指定的开始时间开始计时,而不是从执行时间开始计时。

但是上面如果用schedule方法,间隔时间是3分钟,指定开始时间是2005/12/30 14:10:00,那么在14:17:00分执行这个程序,则立即执行程序一次。并且下一次的执行时间是 14:20,而不是从14:10开始算的周期(14:19)。


如果我想每个月第二个周日的早上8点执行任务,如何?

找quartz 它是一个关于调度任务的库

我也没有用过它,需要的时候去学就ok。

对于这些东西,我们不必要都知道,我们只需要知道在我们碰上某些问题时该去哪里找答案即可。

世界上有千万种药,我们在一生中会得几百几千种病,我们不可能记住所有药的使用方法;

最好的情况是我们知道当我们患了某种病时该去吃某种药。拿到药后慢慢学习它的用法就OK了。

怕就怕,当你得病的时候,你根本不知道该吃哪种药!


参考资料

http://batitan.iteye.com/blog/253483
http://blog.csdn.net/weidan1121/article/details/527307
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值