您好,登錄后才能下訂單哦!
這篇文章給大家分享的是有關Java多線程定時器Timer原理的示例分析的內容。小編覺得挺實用的,因此分享給大家做個參考,一起跟隨小編過來看看吧。
Timer的schedule(TimeTask task, Date time)的使用
該方法的作用是在執行的日期執行一次任務
1、執行任務的時間晚于當前時間:未來執行
private static Timer timer = new Timer(); static public class MyTask extends TimerTask { public void run() { System.out.println("運行了!時間為:" + new Date()); } } public static void main(String[] args) throws Exception { MyTask task = new MyTask(); SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); String dateString = "2015-10-6 12:14:00"; Date dateRef = sdf.parse(dateString); System.out.println("字符串時間:" + dateRef.toLocaleString() + " 當前時間:" + new Date().toLocaleString()); timer.schedule(task, dateRef); }
看一下運行效果:
字符串時間:2015-10-6 12:14:00 當前時間:2015-10-6 12:13:23 運行了!時間為:Tue Oct 06 12:14:00 CST 2015
執行時間和但前時間不一致,而是和dateRef的時間一直,證明了未來執行。任務雖然執行完了,但進程沒有銷毀,控制臺上的方框可以看到還是紅色的,看下Timer的源代碼:
public Timer() { this("Timer-" + serialNumber()); }
public Timer(String name) { thread.setName(name); thread.start(); }
所以,啟動一個Timer就是啟動一個新線程,但是這個新線程并不是守護線程,所以它會一直運行。要運行完就讓進程停止的話,設置Timer為守護線程就好了,有專門的構造函數可以設置:
public Timer(boolean isDaemon) { this("Timer-" + serialNumber(), isDaemon); }
public Timer(String name, boolean isDaemon) { thread.setName(name); thread.setDaemon(isDaemon); thread.start(); }
2、計劃時間早于當前時間:立即執行
如果執行任務的時間早于當前時間,那么立即執行task的任務:
private static Timer timer = new Timer(); static public class MyTask extends TimerTask { public void run() { System.out.println("運行了!時間為:" + new Date()); } } public static void main(String[] args) throws Exception { MyTask task = new MyTask(); SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); String dateString = "2014-10-6 12:14:00"; Date dateRef = sdf.parse(dateString); System.out.println("字符串時間:" + dateRef.toLocaleString() + " 當前時間:" + new Date().toLocaleString()); timer.schedule(task, dateRef); }
看一下運行效果:
字符串時間:2014-10-6 12:14:00 當前時間:2015-10-6 12:20:10 運行了!時間為:Tue Oct 06 12:20:10 CST 2015
執行時間和當前時間一致,證明了立即執行
3、多個TimerTask任務執行
Timer中允許有多個任務:
private static Timer timer = new Timer(); static public class MyTask extends TimerTask { public void run() { System.out.println("運行了!時間為:" + new Date()); } } public static void main(String[] args) throws Exception { MyTask task1 = new MyTask(); MyTask task2 = new MyTask(); SimpleDateFormat sdf1 = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); SimpleDateFormat sdf2 = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); String dateString1 = "2015-10-6 12:26:00"; String dateString2 = "2015-10-6 12:27:00"; Date dateRef1 = sdf1.parse(dateString1); Date dateRef2 = sdf2.parse(dateString2); System.out.println("字符串時間:" + dateRef1.toLocaleString() + " 當前時間:" + new Date().toLocaleString()); System.out.println("字符串時間:" + dateRef2.toLocaleString() + " 當前時間:" + new Date().toLocaleString()); timer.schedule(task1, dateRef1); timer.schedule(task2, dateRef2); }
看一下運行結果:
字符串時間:2015-10-612:26:00當前時間:2015-10-612:25:38
字符串時間:2015-10-612:27:00當前時間:2015-10-612:25:38
運行了!時間為:TueOct0612:26:00CST2015
運行了!時間為:TueOct0612:27:00CST2015
可以看到,運行時間和設置的時間一致,證明了未來可以執行多個任務。另外注意,Task是以隊列的方式一個一個被順序執行的,所以執行的時間有可能和預期的時間不一致,因為前面的任務可能消耗過長,后面任務的運行時間也有可能被延遲。
代碼就不寫了,舉個例子,任務1計劃12:00:00被執行,任務2計劃12:00:10被執行,結果任務1執行了30秒,那么任務2將在12:00:30被執行,因為Task是被放入隊列中的,因此必須一個一個順序運行。
Timer的schedule(TimerTasktask,DatefirstTime,longperiod)
該方法的作用是在指定的日期之后,按指定的間隔周期性地無限循環地執行某一人物
1、計劃時間晚于當前時間:未來執行
static public class MyTask extends TimerTask { public void run() { System.out.println("運行了!時間為:" + new Date()); } } public static void main(String[] args) throws Exception { MyTask task = new MyTask(); SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); String dateString = "2015-10-6 18:00:00"; Timer timer = new Timer(); Date dateRef = sdf.parse(dateString); System.out.println("字符串時間:" + dateRef.toLocaleString() + " 當前時間:" + new Date().toLocaleString()); timer.schedule(task, dateRef, 4000); }
看一下運行結果:
字符串時間:2015-10-6 18:01:00 當前時間:2015-10-6 18:00:15
運行了!時間為:Tue Oct 06 18:01:00 CST 2015
運行了!時間為:Tue Oct 06 18:01:04 CST 2015
運行了!時間為:Tue Oct 06 18:01:08 CST 2015
運行了!時間為:Tue Oct 06 18:01:12 CST 2015
...
看到從設定的時間開始,每隔4秒打印一次,無限打印下去
2、計劃時間早于當前時間:立即執行
static public class MyTask extends TimerTask { public void run() { System.out.println("運行了!時間為:" + new Date()); } } public static void main(String[] args) throws Exception { MyTask task = new MyTask(); SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); String dateString = "2014-10-6 18:01:00"; Timer timer = new Timer(); Date dateRef = sdf.parse(dateString); System.out.println("字符串時間:" + dateRef.toLocaleString() + " 當前時間:" + new Date().toLocaleString()); timer.schedule(task, dateRef, 4000); }
看一下運行結果:
字符串時間:2014-10-6 18:01:00 當前時間:2015-10-6 18:02:46
運行了!時間為:Tue Oct 06 18:02:46 CST 2015
運行了!時間為:Tue Oct 06 18:02:50 CST 2015
運行了!時間為:Tue Oct 06 18:02:54 CST 2015
運行了!時間為:Tue Oct 06 18:02:58 CST 2015
運行了!時間為:Tue Oct 06 18:03:02 CST 2015
...
看到運行時間比當前時間早,從當前時間開始,每隔4秒打印一次,無限循環下去
TimerTask的cancel()方法
TimerTask的cancel()方法的作用是將自身從任務隊列中清除:
static public class MyTaskA extends TimerTask { public void run() { System.out.println("A運行了!時間為:" + new Date()); this.cancel(); } } static public class MyTaskB extends TimerTask { public void run() { System.out.println("B運行了!時間為:" + new Date()); } } public static void main(String[] args) throws Exception { MyTaskA taskA = new MyTaskA(); MyTaskB taskB = new MyTaskB(); SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); String dateString = "2015-10-6 18:10:00"; Timer timer = new Timer(); Date dateRef = sdf.parse(dateString); System.out.println("字符串時間:" + dateRef.toLocaleString() + " 當前時間:" + new Date().toLocaleString()); timer.schedule(taskA, dateRef, 4000); timer.schedule(taskB, dateRef, 4000); }
看一下運行結果:
字符串時間:2015-10-6 18:10:00 當前時間:2015-10-6 18:09:47 A運行了!時間為:Tue Oct 06 18:10:00 CST 2015 B運行了!時間為:Tue Oct 06 18:10:00 CST 2015 B運行了!時間為:Tue Oct 06 18:10:04 CST 2015 B運行了!時間為:Tue Oct 06 18:10:08 CST 2015 B運行了!時間為:Tue Oct 06 18:10:12 CST 2015 ...
看到TimeTask的cancel()方法是將自身從任務隊列中被移除,其他任務不受影響
Timer的cancel()方法
把上面代碼改動一下:
private static Timer timer = new Timer(); static public class MyTaskA extends TimerTask { public void run() { System.out.println("A運行了!時間為:" + new Date()); timer.cancel(); } } static public class MyTaskB extends TimerTask { public void run() { System.out.println("B運行了!時間為:" + new Date()); } } public static void main(String[] args) throws Exception { MyTaskA taskA = new MyTaskA(); MyTaskB taskB = new MyTaskB(); SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); String dateString = "2015-10-6 18:10:00"; Date dateRef = sdf.parse(dateString); System.out.println("字符串時間:" + dateRef.toLocaleString() + " 當前時間:" + new Date().toLocaleString()); timer.schedule(taskA, dateRef, 4000); timer.schedule(taskB, dateRef, 4000); }
看一下運行結果:
字符串時間:2015-10-618:10:00當前時間:2015-10-618:14:15
A運行了!時間為:TueOct0618:14:15CST2015
全部任務都被清除,并且進程被銷毀。不過注意一下,cancel()方法未必一定會停止執行計劃任務,可能正常執行,因為cancel()方法會嘗試去獲取queue鎖,如果并沒有獲取到queue鎖的話,TimerTask類中的任務繼續執行也是完全有可能的
其他方法
再列舉一些Timer中的其他schedule的重載方法的作用,就不提供證明的代碼了,可以自己嘗試一下:
1、schedule(TimerTasktask,longdelay)
以當前時間為參考,在此時間基礎上延遲指定的毫秒數后執行一次TimerTask任務
2、schedule(TimerTasktask,longdelay,longperiod)
以當前時間為參考,在此時間基礎上延遲指定的毫秒數后,以period為循環周期,循環執行TimerTask任務
3、scheduleAtFixedRate(TimerTasktask,DatefirstTime,longperiod)
在延時的場景下,schedule方法和scheduleAtFixedRate方法沒有區別,它們的區別只是在非延時上。如果執行任務的時間沒有被延時,對于schedule方法來說,下一次任務執行的時間參考的是上一次任務的開始時間來計算的;對于scheduleAtFixedRate方法來說,下一次任務執行的時間參考的是上一次任務的結束時間來計算的
感謝各位的閱讀!關于“Java多線程定時器Timer原理的示例分析”這篇文章就分享到這里了,希望以上內容可以對大家有一定的幫助,讓大家可以學到更多知識,如果覺得文章不錯,可以把它分享出去讓更多的人看到吧!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。