Java多線程案例實戰(zhàn)之定時器的實現(xiàn)
一、Timer定時器
Java中,Timer類是用于計劃和執(zhí)行重復(fù)任務(wù)的類(Java標(biāo)準(zhǔn)庫中確實提供了java.util.Timer類)。它可以在指定的時間間隔內(nèi)重復(fù)執(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)部類可以被認(rèn)為是繼承了TimerTask抽象類,并提供了具體的實現(xiàn)代碼。

調(diào)用timer.schedule()方法注冊的任務(wù),會由Timer內(nèi)部的線程池去執(zhí)行,而不是由調(diào)用schedule()方法的線程直接執(zhí)行run()方法。
Timer類內(nèi)部創(chuàng)建了一個線程池,用于執(zhí)行注冊的定時任務(wù)。當(dāng)調(diào)用schedule()方法后,Timer會將傳入的TimerTask對象添加到線程池中進(jìn)行調(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é)束進(jìn)程,原因如下:Timer內(nèi)部有自己的線程,為了保證隨時處理新安排的任務(wù),此線程會一直持續(xù)的執(zhí)行,即此線程影響了阻止來整個進(jìn)程的結(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類中還需要一個掃描線程,一方面要負(fù)責(zé)檢查隊首元素是否是此時應(yīng)該被執(zhí)行的。
// 另一方面,當(dāng)任務(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()) {
// 如果當(dāng)前時間晚于任務(wù)的執(zhí)行時間,就意味著我們要執(zhí)行這個任務(wù)了
queue.poll();
task.getRunnable().run(); // 至此就可以執(zhí)行該任務(wù)了
} else {
// 如果當(dāng)前時間早于任務(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ù)的時間,當(dāng)執(zhí)行時間到的時候,Timer類內(nèi)部的線程就會負(fù)責(zé)調(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?的遠(yuǎn)程代碼執(zhí)行漏洞細(xì)節(jié)被公開,由于?Log4j?的廣泛使用,該漏洞一旦被攻擊者利用會造成嚴(yán)重危害,下面小編給大家?guī)砹薒og4j2?重大漏洞編譯好的log4j-2.15.0.jar包下載,感興趣的朋友一起看看吧2021-12-12
線上dubbo線程池耗盡CyclicBarrier線程屏障異常解決記錄
系統(tǒng)相關(guān)使用人員反饋系統(tǒng)故障,這篇文章主要介紹了線上dubbo線程池耗盡CyclicBarrier線程屏障異常解決的記錄,有需要的朋友可以借鑒參考下2022-03-03
java訪問者模式的靜態(tài)動態(tài)及偽動態(tài)分派徹底理解
這篇文章主要為大家介紹了java訪問者模式的靜態(tài)動態(tài)及偽動態(tài)分派徹底理解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-06-06
Java實現(xiàn)在PPT中創(chuàng)建SmartArt圖形的示例代碼
SmartArt其實就是一個文字的可視化工具,用戶可在PowerPoint,Word,Excel中使用該特性創(chuàng)建各種圖形圖表。本文就將為您介紹如何通過Java應(yīng)用程序在PPT中創(chuàng)建SmartArt圖形,需要的可以參考一下2023-04-04

