zoukankan      html  css  js  c++  java
  • Java中使用Timer和TimerTask实现多线程

          Timer是一种线程设施,用于安排以后在后台线程中执行的任务。可安排任务执行一次,或者定期重复执行,可以看成一个定时器,可以调度TimerTask。TimerTask是一个抽象类,实现了Runnable接口,所以具备了多线程的能力。

    测试代码:

    import java.util.TimerTask;
      
    public class OneTask extends TimerTask{
      
        private int id;
        public OneTask(int id){
            this.id = id;
        }
          
        @Override
        public void run() {
            System.out.println("线程"+ id +":  正在 执行。。"); 
            //System.gc();
        }   
    }

    然后主程序代码为:

    import java.util.Date;
    import java.util.Timer;
       
    public class Test1 {
           
        public static void main(String[] args) {
            Timer timer = new Timer(); 
      
            timer.schedule(new OneTask(1), 5000);// 5秒后启动任务
              
            OneTask secondTask= new OneTask(2);
            timer.schedule(secondTask, 1000, 3000);// 1秒后启动任务,以后每隔3秒执行一次线程
              
            Date date = new Date();
            timer.schedule(new OneTask(3),new Date(date.getTime()+1000));//以date为参数,指定某个时间点执行线程
              
    //      timer.cancel();
    //      secondTask.cancel();
            System.out.println("end in main thread...");
        }
    }



    Timer里面有4个schedule重载函数。而且还有两个scheduleAtFixedRate:

    void scheduleAtFixedRate(TimerTask task, Date firstTime, long period)
    安排指定的任务在指定的时间开始进行重复的固定速率执行。
    void scheduleAtFixedRate(TimerTask task, long delay, long period)
    安排指定的任务在指定的延迟后开始进行重复的固定速率执行。

    使用scheduleAtFixedRate的话, Timer会尽量的让任务在一个固定的频率下运行。例如:在上面的例子中,让secondTask在1秒钟后,每3秒钟执行一次,但是因为java不是实时的,所以,我们在上个程序中表达的原义并不能够严格执行,例如有时可能资源调度紧张4秒以后才执行下一次,有时候又3.5秒执行。如果我们调用的是scheduleAtFixedRate,那么Timer会尽量让你的secondTask执行的频率保持在3秒一次。运行上面的程序,假设使用的是scheduleAtFixedRate,那么下面的场景就是可能的:1秒钟后,secondTask执行一次,因为系统繁忙,之后的3.5秒后secondTask才得以执行第二次,然后Timer记下了这个延迟,并尝试在下一个任务的时候弥补这个延迟,那么2.5秒后,secondTask 将执行的三次。“以固定的频率而不是固定的延迟时间去执行一个任务”就是这个意思。

    Timer终止的问题:

    默认情况下,只要一个程序的timer线程在运行,那么这个程序就会保持运行。可以通过以下3种方法终止一个timer线程:
    (1)调用timer的cancle方法。你可以从程序的任何地方调用此方法,甚至在一个timer task的run方法里;
    (2)让timer线程成为一个daemon线程(可以在创建timer时使用new Timer(true)达到这个目地),这样当程序只有daemon线程的时候,它就会自动终止运行;
    (3)调用System.exit方法,使整个程序(所有线程)终止。

    TimerTask也有cancel方法。

    上面所说的“只要一个程序的timer线程在运行,那么这个程序就会保持运行”。那么反过来,如果Timer里的所有TimerTask都执行完了,整个程序会退出吗,经测试答案是否定的,例如上面的测试代码,如果只加第一个TimerTask在Timer中执行:

    timer.schedule(new OneTask(1), 5000);// 5秒后启动任务
    那么5秒以后,其实整个程序还是没有退出,Timer会等待垃圾回收的时候被回收掉然后程序会得以退出,但是多长时间呢?

    在TimerTask的run函数执行完以后加上System.gc();就可以了。

    原文摘自  http://www.bdqn.cn/news/201305/9303.shtml

  • 相关阅读:
    poj 1035 字符串匹配
    拓扑排序的小总结
    POJ1018
    POJ1328详细题解
    POJ1159题解报告
    POJ1088 (滑雪)
    树状树组区间修改,单点修改模板
    spfa模板
    树状树组离散化求逆序对模板
    POJ3723(最小生成树,负权)
  • 原文地址:https://www.cnblogs.com/honey01/p/6744214.html
Copyright © 2011-2022 走看看