java timer和timertask_java定时器Timer和TimerTask详解
目錄結(jié)構(gòu):
Timer和TimerTask
一個(gè)Timer調(diào)度的例子
如何終止Timer線程
關(guān)于cancle方式終止線程
反復(fù)執(zhí)行一個(gè)任務(wù)
schedule VS.?scheduleAtFixedRate
一些注意點(diǎn)
1. Timer和TimerTask
Timer是jdk中提供的一個(gè)定時(shí)器工具,使用的時(shí)候會(huì)在主線程之外起一個(gè)單獨(dú)的線程執(zhí)行指定的計(jì)劃任務(wù),可以指定執(zhí)行一次或者反復(fù)執(zhí)行多次。
TimerTask是一個(gè)實(shí)現(xiàn)了Runnable接口的抽象類,代表一個(gè)可以被Timer執(zhí)行的任務(wù)。
2. 一個(gè)Timer調(diào)度的例子
1 importjava.util.Timer;2 importjava.util.TimerTask;3
4 public classTestTimer {5
6 public static voidmain(String args[]){7 System.out.println("About to schedule task.");8 new Reminder(3);9 System.out.println("Task scheduled.");10 }11
12 public static classReminder{13 Timer timer;14
15 public Reminder(intsec){16 timer = newTimer();17 timer.schedule(newTimerTask(){18 public voidrun(){19 System.out.println("Time's up!");20 timer.cancel();21 }22 }, sec*1000);23 }24 }
25 }
運(yùn)行之后,在console會(huì)首先看到:
About to schedule task.Task scheduled.
然后3秒鐘后,看到
Time's up!
從這個(gè)例子可以看出一個(gè)典型的利用timer執(zhí)行計(jì)劃任務(wù)的過程如下:
new一個(gè)TimerTask的子類,重寫run方法來指定具體的任務(wù),在這個(gè)例子里,我用匿名內(nèi)部類的方式來實(shí)現(xiàn)了一個(gè)TimerTask的子類
new一個(gè)Timer類,Timer的構(gòu)造函數(shù)里會(huì)起一個(gè)單獨(dú)的線程來執(zhí)行計(jì)劃任務(wù)。jdk的實(shí)現(xiàn)代碼如下:
1 publicTimer() {2 this("Timer-" +serialNumber());3 }4
5 publicTimer(String name) {6 thread.setName(name);7 thread.start();8 }
調(diào)用相關(guān)調(diào)度方法執(zhí)行計(jì)劃。這個(gè)例子調(diào)用的是schedule方法。
任務(wù)完成,結(jié)束線程。這個(gè)例子是調(diào)用cancel方法結(jié)束線程。
3. 如何終止Timer線程
默認(rèn)情況下,創(chuàng)建的timer線程會(huì)一直執(zhí)行,主要有下面四種方式來終止timer線程:
調(diào)用timer的cancle方法
把timer線程設(shè)置成daemon線程,(new Timer(true)創(chuàng)建daemon線程),在jvm里,如果所有用戶線程結(jié)束,那么守護(hù)線程也會(huì)被終止,不過這種方法一般不用。
當(dāng)所有任務(wù)執(zhí)行結(jié)束后,刪除對(duì)應(yīng)timer對(duì)象的引用,線程也會(huì)被終止。
調(diào)用System.exit方法終止程序
4. 關(guān)于cancle方式終止線程
這種方式終止timer線程,jdk的實(shí)現(xiàn)比較巧妙,稍微說一下。
首先看cancle方法的源碼:
1 public voidcancel() {2 synchronized(queue) {3 thread.newTasksMayBeScheduled = false;4 queue.clear();5 queue.notify(); //In case queue was already empty.
6 }7 }
沒有顯式的線程stop方法,而是調(diào)用了queue的clear方法和queue的notify方法,clear是個(gè)自定義方法,notify是Objec自帶的方法,很明顯是去喚醒wait方法的。
再看clear方法:
1 voidclear() {2 //Null out task references to prevent memory leak
3 for (int i=1; i<=size; i++)4 queue[i] = null;5
6 size = 0;7 }
clear方法很簡單,就是去清空queue,queue是一個(gè)TimerTask的數(shù)組,然后把queue的size重置成0,變成empty.還是沒有看到顯式的停止線程方法,回到最開始new Timer的時(shí)候,看看new Timer代碼:
1 publicTimer() {2 this("Timer-" +serialNumber());3 }4
5 publicTimer(String name) {6 thread.setName(name);7 thread.start();8 }
看看這個(gè)內(nèi)部變量thread:
1 /**
2 * The timer thread.3 */
4 private TimerThread thread = new TimerThread(queue);
不是原生的Thread,是自定義的類TimerThread.這個(gè)類實(shí)現(xiàn)了Thread類,重寫了run方法,如下:
1 public voidrun() {2 try{3 mainLoop();4 } finally{5 //Someone killed this Thread, behave as if Timer cancelled
6 synchronized(queue) {7 newTasksMayBeScheduled = false;8 queue.clear(); //Eliminate obsolete references
9 }10 }11 }
最后是這個(gè)mainLoop方法,這方法比較長,截取開頭一段:
1 private voidmainLoop() {2 while (true) {3 try{4 TimerTask task;5 booleantaskFired;6 synchronized(queue) {7 //Wait for queue to become non-empty
8 while (queue.isEmpty() &&newTasksMayBeScheduled)9 queue.wait();10 if(queue.isEmpty())11 break; //Queue is empty and will forever remain; die
可以看到wait方法,之前的notify就是通知到這個(gè)wait,然后clear方法在notify之前做了清空數(shù)組的操作,所以會(huì)break,線程執(zhí)行結(jié)束,退出。
5. 反復(fù)執(zhí)行一個(gè)任務(wù)
通過調(diào)用三個(gè)參數(shù)的schedule方法實(shí)現(xiàn),最后一個(gè)參數(shù)是執(zhí)行間隔,單位毫秒。
6. schedule VS.?scheduleAtFixedRate
這兩個(gè)方法都是任務(wù)調(diào)度方法,他們之間區(qū)別是,schedule會(huì)保證任務(wù)的間隔是按照定義的period參數(shù)嚴(yán)格執(zhí)行的,如果某一次調(diào)度時(shí)間比較長,那么后面的時(shí)間會(huì)順延,保證調(diào)度間隔都是period,而scheduleAtFixedRate是嚴(yán)格按照調(diào)度時(shí)間來的,如果某次調(diào)度時(shí)間太長了,那么會(huì)通過縮短間隔的方式保證下一次調(diào)度在預(yù)定時(shí)間執(zhí)行。舉個(gè)栗子:你每個(gè)3秒調(diào)度一次,那么正常就是0,3,6,9s這樣的時(shí)間,如果第二次調(diào)度花了2s的時(shí)間,如果是schedule,就會(huì)變成0,3+2,8,11這樣的時(shí)間,保證間隔,而scheduleAtFixedRate就會(huì)變成0,3+2,6,9,壓縮間隔,保證調(diào)度時(shí)間。
7. 一些注意點(diǎn)
每一個(gè)Timer僅對(duì)應(yīng)唯一一個(gè)線程。
Timer不保證任務(wù)執(zhí)行的十分精確。
Timer類的線程安全的。
總結(jié)
以上是生活随笔為你收集整理的java timer和timertask_java定时器Timer和TimerTask详解的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 最新RiPro6.3.8 wordpre
- 下一篇: 论图像数据标注