From Gossip@Openhome

Java Gossip: TimerTask 與 Timer

如果您想要在某個時間點之後,執行某項排定的任務,您可以定義一個TimerTask,再將之排定給Timer,Timer會依指定的時間來執行TimerTask。

舉個實例來說,您想在程式啟動後5秒鐘執行報時,則您可以繼承TimerTask,並重新定義其run()方法。

  • DateTask.java
package onlyfun.caterpillar;

import java.util.*;

public class DateTask extends TimerTask {
public void run() {
System.out.println("任務時間:" + new Date());
}
}

在run()方法中,只簡單的顯示run()方法被執行的時間,TimerTask的排程執行要交給Timer物件,例如:
  • Main.java
package onlyfun.caterpillar;

import java.util.*;

public class Main {
public static void main(String[] args) {
Timer timer = new Timer();
timer.schedule(new DateTask(), 5000);
System.out.println("現在時間:" + new Date());

try {
Thread.sleep(8000);
}
catch(InterruptedException e) {
}

timer.cancel();
}
}

Timer物件的schedule()方法接受TimerTask實例,並可以給它一個毫秒數,表示schedule()方法執行後,幾毫秒之後執行 TimerTask中的run()方法,Timer物件在啟動排程後,如果要停止,則要執行cancel()方法,這個程式的執行結果如下:

現在時間:Fri Jan 19 10:47:10 CST 2007
任務時間:Fri Jan 19 10:47:15 CST 2007


除了指定幾毫秒之後執行任務之外,您還可以要求Timer物件,每隔多少時間就執行一次TimerTask的run()方法,例如一秒後開始執行TimerTask,而後每隔3秒執行一次:
  • Main.java
package onlyfun.caterpillar;

import java.util.*;

public class Main {
public static void main(String[] args) {
Timer timer = new Timer();
timer.schedule(new DateTask(), 1000, 3000);
System.out.println("現在時間:" + new Date());

try {
Thread.sleep(20000);
}
catch(InterruptedException e) {
}

timer.cancel();
}
}

執行結果如下:
現在時間:Fri Jan 19 10:49:43 CST 2007
任務時間:Fri Jan 19 10:49:44 CST 2007
任務時間:Fri Jan 19 10:49:47 CST 2007
任務時間:Fri Jan 19 10:49:50 CST 2007
任務時間:Fri Jan 19 10:49:53 CST 2007
任務時間:Fri Jan 19 10:49:56 CST 2007
任務時間:Fri Jan 19 10:49:59 CST 2007
任務時間:Fri Jan 19 10:50:02 CST 2007

Timer的schedule()方法也可以接受Date實例,例如您想直接排定執行日為一個月之後,則可以透過Calendar來計算,再取得Date時間:
        Calendar c = Calendar.getInstance();
        c.add(Calendar.MONTH, 1);
        System.out.println("任務排定時間:" + c.getTime());

        Timer timer = new Timer();
        timer.schedule(new DateTask(), c.getTime(), 3000);

Timer與TimerTask是獨立的兩個物件,當使用Timer的schedule()方法排定TimerTask之後,則必須等執行Timer的 cancel()執行之後,讓TimerTask與Timer脫離關係,TimerTask才可以重新加入其它Timer的排程。