Timer的用法
使用Timer和TimerTask组合
最后一种实现多线程的方式,就是使用java.util包中的Timer和TimerTask类实现多线程,使用这种方式也可以比较方便的实现线程。
在这种实现方式中,Timer类实现的是类似闹钟的功能,也就是定时或者每隔一定时间触发一次线程。其实,Timer类本身实现的就是一个线程,只是这个线程是用来实现调用其它线程的。而TimerTask类是一个抽象类,该类实现了Runnable接口,所以按照前面的介绍,该类具备多线程的能力。
在这种实现方式中,通过继承TimerTask使该类获得多线程的能力,将需要多线程执行的代码书写在run方法内部,然后通过Timer类启动线程的执行。
在实际使用时,一个Timer可以启动任意多个TimerTask实现的线程,但是多个线程之间会存在阻塞。所以如果多个线程之间如果需要完全独立运行的话,最好还是一个Timer启动一个TimerTask实现。
使用该种实现方式实现的多线程示例代码如下:
import java.util.*;
/**
* 测试类
*/
public class Test3 {
public static void main(String[] args) {
//创建Timer
Timer t = new Timer();
//创建TimerTask
MyTimerTask mtt1 = new MyTimerTask("线程1:");
//启动线程
t.schedule(mtt1, 0);
}
}
import java.util.TimerTask;
/**
* 以继承TimerTask类的方式实现多线程
*/
public class MyTimerTask extends TimerTask {
String s;
public MyTimerTask(String s){
this.s = s;
}
public void run() {
try{
for(int i = 0;
i < 10;
i++){
Thread.sleep(1000);
System.out.println(s + i);
}
}catch(Exception e){}
}
}
在该示例中,MyTimerTask类实现了多线程,以多线程方式执行的代码书写在该类的run方法内部,该类的功能和前面的多线程的代码实现类似。
而在该代码中,启动线程时需要首先创建一个Timer类的对象,以及一个MyTimerTask线程类的兑现,然后使用Timer对象的schedule方法实现,启动线程的代码为:
//创建Timer
Timer t = new Timer();
//创建TimerTask
MyTimerTask mtt1 = new MyTimerTask("线程1:");
//启动线程
t.schedule(mtt1, 0);
其中schedule方法中的第一个参数mtt1代表需要启动的线程对象,而第二个参数0则代表延迟0毫秒启动该线程,也就是立刻启动。
由于schedule方法比较重要,下面详细介绍一下Timer类中的四个schedule方法:
1、 public void schedule(TimerTask task,Date time)
该方法的作用是在到达time指定的时间或已经超过该时间时执行线程task。例如假设t是Timer对象,task是需要启动的TimerTask线程对象,后续示例也采用这种约定实现,则启动线程的示例代码如下:
Date d = new Date(2009-1900,10-1,1,10,0,0);
t. schedule(task,d);
则该示例代码的作用是在时间达到d指定的时间或超过该时间(例如2009年10月2号)时,启动线程task。
2、 public void schedule(TimerTask task, Date firstTime, long period)
该方法的作用是在时间到达firstTime开始,每隔period毫秒就启动一次task指定的线程。示例代码如下:
Date d = new Date(2009-1900,10-1,1,10,0,0);
t. schedule(task,d,20000);
该示例代码的作用是当时间达到或超过d指定的时间以后,每隔20000毫秒就启动一次线程task,这种方式会重复触发线程。
3、 public void schedule(TimerTask task,long delay)
该方法和第一个方法类似,作用是在执行schedule方法以后delay毫秒以后启动线程task。示例代码如下:
t. schedule(task,1000);
该示例代码的作用是在执行该行启动代码1000毫秒以后启动一次线程task。
4、 public void schedule(TimerTask task,long delay,long period)
该方法和第二个方法类似,作用是在执行schedule方法以后delay毫秒以后启动线程task,然后每隔period毫秒重复启动线程task。
例外需要说明的是Timer类中启动线程还包含两个scheduleAtFixedRate方法,这两个方法的参数和上面的第二个和第四个一致,其作用是实现重复启动线程时的精确延时。对于schedule方法来说,如果重复的时间间隔是1000毫秒,则实际的延迟时间是1000毫秒加上系统执行时消耗的时间,例如为5毫秒,则实际每轮的时间间隔为1005毫秒。而对于scheduleAtFixedRate方法来说,如果设置的重复时间间隔为1000毫秒,系统执行时消耗的时间为5毫秒,则延迟时间就会变成995毫秒,从而保证每轮间隔为1000毫秒。
介绍完了schedule方法以后,让我们再来看一下前面的示例代码,如果在测试类中启动两个MyTimerTask线程,一种实现的代码为:
import java.util.Timer;
/**
* 测试类
*/
public class Test4 {
public static void main(String[] args) {
//创建Timer
Timer t = new Timer();
//创建TimerTask
MyTimerTask mtt1 = new MyTimerTask("线程1:");
MyTimerTask mtt2 = new MyTimerTask("线程2:");
//启动线程
System.out.println("开始启动");
t.schedule(mtt1, 1000);
System.out.println("启动线程1");
t.schedule(mtt2, 1000);
System.out.println("启动线程2");
}
}
在该示例代码中,使用一个Timer对象t依次启动了两个MyTimerTask类型的对象mtt1和mtt2。而程序的执行结果是:
开始启动
启动线程1
启动线程2
线程1:0
线程1:1
线程1:2
线程1:3
线程1:4
线程1:5
线程1:6
线程1:7
线程1:8
线程1:9
线程2:0
线程2:1
线程2:2
线程2:3
线程2:4
线程2:5
线程2:6
线程2:7
线程2:8
线程2:9
从程序的执行结果可以看出,在Test4类中mtt1和mtt2都被启动,按照前面的schedule方法介绍,这两个线程均会在线程启动以后1000毫秒后获得执行。但是从实际执行效果却可以看出这两个线程不是同时执行的,而是依次执行,这主要是因为一个Timer启动的多个TimerTask之间会存在影响,当上一个线程未执行完成时,会阻塞后续线程的执行,所以当线程1执行完成以后线程2才获得了执行。
如果需要线程1和线程2获得同时执行,则只需要分别使用两个Timer启动TimerTask线程即可,启动的示例代码如下:
import java.util.Timer;
/**
* 测试类
*/
public class Test5 {
public static void main(String[] args) {
//创建Timer
Timer t1 = new Timer();
Timer t2 = new Timer();
//创建TimerTask
MyTimerTask mtt1 = new MyTimerTask("线程1:");
MyTimerTask mtt2 = new MyTimerTask("线程2:");
//启动线程
System.out.println("开始启动");
t1.schedule(mtt1, 1000);
System.out.println("启动线程1");
t2.schedule(mtt2, 1000);
System.out.println("启动线程2");
}
}
在该示例中,分别使用两个Timer对象t1和t2,启动两个TimerTask线程对象mtt1和mtt2,两者之间不互相干扰,所以达到了同时执行的目的。
在使用上面的示例进行运行时,由于Timer自身的线程没有结束,所以在程序输出完成以后程序还没有结束,需要手动结束程序的执行。例如在Eclipse中可以点击控制台上面的红色“Teminate”按钮结束程序。
推荐阅读
- 热闹中的孤独
- JAVA(抽象类与接口的区别&重载与重写&内存泄漏)
- 放屁有这三个特征的,请注意啦!这说明你的身体毒素太多
- 一个人的旅行,三亚
- 布丽吉特,人生绝对的赢家
- 慢慢的美丽
- 尽力
- 一个小故事,我的思考。
- 家乡的那条小河
- 《真与假的困惑》???|《真与假的困惑》??? ——致良知是一种伟大的力量