Java 中 Timer 和 TimerTask實現多線程

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, 10003000);// 1秒後啓動任務,以後每隔3秒執行一次線程
          
        Date date = new Date();
        timer.schedule(new OneTask(3),newDate(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();就可以了。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章