Java多線程案例實戰(zhàn)之定時器的實現(xiàn)
一、Timer定時器
Java中,Timer類
是用于計劃和執(zhí)行重復任務(wù)的類(Java標準庫中確實提供了java.util.Timer
類)。它可以在指定的時間間隔內(nèi)重復執(zhí)行一個任務(wù),或者在指定時間點執(zhí)行任務(wù)。
二、Timer定時器的設(shè)計
選擇java.util
包中的Timer
類:
使用了Timer類的schedule()
方法來安排一個任務(wù)在延遲3000毫秒后執(zhí)行。在TimerTask的run()方法中,我們編寫需要執(zhí)行的具體任務(wù)邏輯。我們現(xiàn)在來了解一下TimerTask()這個抽象類
(如下圖):該類是一個抽象類,并且繼承了Runnable方法
。創(chuàng)建了一個匿名內(nèi)部類并實現(xiàn)了run()方法。這個匿名內(nèi)部類可以被認為是繼承了TimerTask抽象類,并提供了具體的實現(xiàn)代碼。
調(diào)用timer.schedule()
方法注冊的任務(wù),會由Timer內(nèi)部的線程池去執(zhí)行,而不是由調(diào)用schedule()方法的線程直接執(zhí)行run()方法。
Timer類內(nèi)部創(chuàng)建了一個線程池,用于執(zhí)行注冊的定時任務(wù)。當調(diào)用schedule()方法后,Timer會將傳入的TimerTask對象添加到線程池中進行調(diào)度。
下面是一個簡單的定時器程序,可以運行試試看:
import java.util.Timer; import java.util.TimerTask; public class Demo22 { public static void main(String[] args) { Timer timer = new Timer(); timer.schedule(new TimerTask() { @Override public void run() { System.out.println("hello world!!!"); } },3000); System.out.println("程序開始執(zhí)行嘍!!!"); } }
運行結(jié)果如下:
可以看到程序并沒有結(jié)束進程,原因如下:Timer
內(nèi)部有自己的線程,為了保證隨時處理新安排的任務(wù),此線程會一直持續(xù)的執(zhí)行,即此線程影響了阻止來整個進程的結(jié)束。
定時器是支持多個任務(wù)同時執(zhí)行的,請看:
三、定時器的實現(xiàn)
代碼實現(xiàn)如下:
import java.util.Comparator; import java.util.PriorityQueue; class MyTimerTask implements Comparable<MyTimerTask> { private long time; // 表示任務(wù)什么時候開始執(zhí)行 private Runnable runnable; // 表示具體任務(wù)是啥 public MyTimerTask(Runnable runnable,long delay) { // delay是一個相對的時間差 time = System.currentTimeMillis() + delay;// 這里計算出任務(wù)執(zhí)行的具體時間 this.runnable = runnable; } public long getTime() { return time; } public Runnable getRunnable() { return runnable; } @Override public int compareTo(MyTimerTask o) { // 時間最少的元素放在隊首,即時間越少優(yōu)先級越高 return (int)(this.time - o.time); // time是long類型 } } // 這是定時器類的本體 class MyTimer { // 使用優(yōu)先級隊列來保存上面的N個任務(wù) private PriorityQueue<MyTimerTask> queue = new PriorityQueue<>(); // locker是用來加鎖的對象 private Object locker = new Object(); // 定時器的核心方法,即把要執(zhí)行的任務(wù)添加到隊列中 public void schedule(Runnable runnable,long delay) { synchronized (locker) { MyTimerTask task = new MyTimerTask(runnable,delay); queue.offer(task); // 每次來新的任務(wù)之后都會喚醒一下掃描線程,此時掃描線程就可以根據(jù)最新的任務(wù)情況來重新規(guī)劃等待時間 locker.notify(); } } // MyTimer類中還需要一個掃描線程,一方面要負責檢查隊首元素是否是此時應(yīng)該被執(zhí)行的。 // 另一方面,當任務(wù)到點開始執(zhí)行之后,需要調(diào)用Runnable中的run方法來完成任務(wù) public MyTimer() { // 掃描線程 Thread t = new Thread(() -> { while(true) { try { synchronized(locker) { while(queue.isEmpty()) { // 隊列為空時,此時不應(yīng)該取這里的元素 locker.wait(); } MyTimerTask task = queue.peek(); long curTime = System.currentTimeMillis(); if(curTime > task.getTime()) { // 如果當前時間晚于任務(wù)的執(zhí)行時間,就意味著我們要執(zhí)行這個任務(wù)了 queue.poll(); task.getRunnable().run(); // 至此就可以執(zhí)行該任務(wù)了 } else { // 如果當前時間早于任務(wù)的執(zhí)行時間,誒呀太早了,讓這個線程(休眠)休息一會一會吧!!! // Thread.sleep(task.getTime() - curTime); locker.wait(task.getTime() - curTime); } } } catch (InterruptedException e) { e.printStackTrace(); } } }); t.start(); } } public class Demo23 { public static void main(String[] args) { MyTimer timer = new MyTimer(); timer.schedule(new Runnable() { @Override public void run() { System.out.println("hello world! 3"); } },3000); timer.schedule(new Runnable() { @Override public void run() { System.out.println("hello world! 2"); } },2000); timer.schedule(new Runnable() { @Override public void run() { System.out.println("hello world! 1"); } },1000); System.out.println("程序開始執(zhí)行!!!"); } }
運行結(jié)果如下:
四、總結(jié)
Timer類是Java中的定時工具類,它可以幫助我們實現(xiàn)在指定時間執(zhí)行指定任務(wù)的功能。Timer類提供了一個方法即schedule方法
,我們可以通過schedule方法來注冊一個任務(wù)并指定執(zhí)行該任務(wù)的時間,當執(zhí)行時間到的時候,Timer類內(nèi)部的線程就會負責調(diào)用執(zhí)行注冊的任務(wù)。
另外我們可以通過優(yōu)先級隊列的方式來實現(xiàn)類似于Timer類這樣的定時器。
到此這篇關(guān)于Java多線程案例定時器的實現(xiàn)的文章就介紹到這了,更多相關(guān)Java定時器實現(xiàn)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Log4j2?重大漏洞編譯好的log4j-2.15.0.jar包下載(替換過程)
Apache?開源項目?Log4j?的遠程代碼執(zhí)行漏洞細節(jié)被公開,由于?Log4j?的廣泛使用,該漏洞一旦被攻擊者利用會造成嚴重危害,下面小編給大家?guī)砹薒og4j2?重大漏洞編譯好的log4j-2.15.0.jar包下載,感興趣的朋友一起看看吧2021-12-12線上dubbo線程池耗盡CyclicBarrier線程屏障異常解決記錄
系統(tǒng)相關(guān)使用人員反饋系統(tǒng)故障,這篇文章主要介紹了線上dubbo線程池耗盡CyclicBarrier線程屏障異常解決的記錄,有需要的朋友可以借鑒參考下2022-03-03java訪問者模式的靜態(tài)動態(tài)及偽動態(tài)分派徹底理解
這篇文章主要為大家介紹了java訪問者模式的靜態(tài)動態(tài)及偽動態(tài)分派徹底理解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-06-06Java實現(xiàn)在PPT中創(chuàng)建SmartArt圖形的示例代碼
SmartArt其實就是一個文字的可視化工具,用戶可在PowerPoint,Word,Excel中使用該特性創(chuàng)建各種圖形圖表。本文就將為您介紹如何通過Java應(yīng)用程序在PPT中創(chuàng)建SmartArt圖形,需要的可以參考一下2023-04-04