schedule和scheduleAtFixedRate區別

schedule和scheduleAtFixedRate的區別在於,如果指定開始執行的時間在當前系統運行時間之前,scheduleAtFixedRate會把已經過去的時間也作爲週期執行,而schedule不會把過去的時間算上。

比如

SimpleDateFormat fTime = new SimpleDateFormat("yyyy/MM/dd HH:mm:ss");
Date d1 = fTime.parse("2005/12/30 14:10:00");

t.scheduleAtFixedRate(new TimerTask(){
public void run()
{
System.out.println("this is task you do6");
}
},d1,3*60*1000);

間隔時間是3分鐘,指定開始時間是2005/12/30 14:10:00,如果我在14:17:00分執行這個程序,那麼會立刻打印3次

this is task you do6 //14:10
this is task you do6 //14:13
this is task you do6 //14:16

並且注意,下一次執行是在14:19 而不是 14:20。就是說是從指定的開始時間開始計時,而不是從執行時間開始計時。

但是上面如果用schedule方法,間隔時間是3分鐘,指定開始時間是2005/12/30 14:10:00,那麼在14:17:00分執行這個程序,則立即執行程序一次。並且下一次的執行時間是 14:20,而不是從14:10開始算的週期(14:19)。

需要注意的是scheduleAtFixedRate和schedule在參數完全相同的情況下,執行效果是不同的。上面例子中任務只是打印一個字符串,比較簡單。但如果任務比較複雜,或者由於任何原因(如垃圾回收或其他後臺活動)而延遲了某次執行,則scheduleAtFixedRate方法將快速連續地出現兩次或更多的執行,從而使後續執行能夠“追趕上來”;而schedule方法的後續執行也將被延遲。所以,在長期運行中,scheduleAtFixedRate執行的頻率將正好是指定週期的倒數,schedule 執行的頻率一般要稍慢於指定週期的倒數。

==================================================================================================

schedule和scheduleAtFixedRate 區別:

(1) 2個參數的schedule在制定任務計劃時,如果指定的計劃執行時間scheduledExecutionTime<= systemCurrentTime,則task會被立即執行。scheduledExecutionTime不會因爲某一個task的過度執行而改變。
(2) 3個參數的schedule在制定反覆執行一個task的計劃時,每一次執行這個task的計劃執行時間隨着前一次的實際執行時間而變,也就是 scheduledExecutionTime(第n+1次)=realExecutionTime(第n次)+periodTime。也就是說如果第n 次執行task時,由於某種原因這次執行時間過長,執行完後的systemCurrentTime>= scheduledExecutionTime(第n+1次),則此時不做時隔等待,立即執行第n+1次task,而接下來的第n+2次task的 scheduledExecutionTime(第n+2次)就隨着變成了realExecutionTime(第n+1次)+periodTime。說白了,這個方法更注重保持間隔時間的穩定。
(3)3個參數的scheduleAtFixedRate在制定反覆執行一個task的計劃時,每一次執行這個task的計劃執行時間在最初就被定下來了,也就是scheduledExecutionTime(第n次)=firstExecuteTime +n*periodTime;如果第n次執行task時,由於某種原因這次執行時間過長,執行完後的systemCurrentTime>= scheduledExecutionTime(第n+1次),則此時不做period間隔等待,立即執行第n+1次task,而接下來的第n+2次的 task的scheduledExecutionTime(第n+2次)依然還是firstExecuteTime+(n+2)*periodTime這在第一次執行task就定下來了。說白了,這個方法更注重保持執行頻率的穩定。

Timer的實例:

package com.hemes.timer;

import java.util.*;

public class doTask extends TimerTask {

// true時使用後臺進程線程。只要剩下的程序記敘運行,後臺進程線程就會執行。
Timer myTimer;

public void start(int delay, int hour) {
myTimer = new Timer();
myTimer.schedule(this, delay * 1000, hour*1000*60*60); //利用timer.schedule方法

//public void schedule(TimerTask task,long time,long period)
//task被安排在延遲time後執行,執行後將每隔period(毫秒)反覆執行。由於規定的時間間隔並不能保證與時鐘精準的同不步,所以該方
}

public void start(Date time, int hour) {
myTimer = new Timer();
myTimer.schedule(this, time, hour*1000*60*60); //利用timer.schedule方法

//public void schedule(TimerTask task,Date time,long period)
//task被安排在time指定的時間執行,執行後將每隔period(毫秒)反覆執行。由於規定的時間間隔並不能保證與時鐘精準的同不步,所以該方
}

public void run() {
//執行任務(sql)
System.out.println("do Task...");
}

public void end(){
myTimer.cancel();
//終止Timer的功能執行,但不會對正在執行的任務有影響。當執行cancel方法後將不能再用其分配任務。
}

//測試
public static void main(String args[]) {

doTask myTask1 = new doTask();

//Get the Date corresponding to 11:30:00 pm today.
Calendar calendar = Calendar.getInstance();
calendar.set(Calendar.HOUR_OF_DAY, 23);
calendar.set(Calendar.MINUTE, 30);
calendar.set(Calendar.SECOND, 0);
Date time = calendar.getTime();

myTask1.start(time,24);



//myTask1.end();//線程結束

}
}

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