Timer是一種線程設施,用於安排以後在後臺線程中執行的任務。可安排任務執行一次,或者定期重複執行,可以看成一個定時器,可以調度TimerTask。TimerTask是一個抽象類,實現了Runnable接口,所以具備了多線程的能力。
測試代碼:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
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(); } } |
然後主程序代碼爲:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
|
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();就可以了。