1. 程式人生 > >不用任何框架,Java 就能實現定時任務的 3 種方法!

不用任何框架,Java 就能實現定時任務的 3 種方法!

是的,不用任何框架,用我們樸素的 Java 程式語言就能實現定時任務。 今天,棧長就介紹 3 種實現方法,教你如何使用 JDK 實現定時任務! ## 1、 sleep 這也是我們最常用的 sleep 休眠大法,不只是當作休眠用,我們還可以利用它很輕鬆的能實現一個簡單的定時任務。 **實現邏輯:** 新開一個執行緒,新增一個 for/ while 死迴圈,然後在死迴圈裡面新增一個 sleep 休眠邏輯,讓程式每隔 N 秒休眠再執行一次,這樣就達到了一個簡單定時任務的效果。 **實現程式碼如下:** ``` /** * 休眠實現定時任務 * 來源公眾號:Java技術棧 */ private static void sleepTask() { new Thread(() -> { while (true) { System.out.println("hi, 歡迎關注:Java技術棧"); try { // 每隔3秒執行一次 Thread.sleep(3000); } catch (InterruptedException e) { e.printStackTrace(); } } }).start(); } ``` 這種方式比較傻瓜化了,只能按固定頻率執行,不能指定具體執行的時間。 另外,上面的箭頭語法,棧長使用了 JDK 8 中的 Lambda 表示式,這裡就不再撰述了,Java 8 系列實戰教程我都寫了一堆了,不清楚的可以關注公眾號:Java技術棧,在後臺回覆 "java" 閱讀,我都整理好了。 ## 2、Timer 來看下 JDK 自帶的 java.util.Timer 類: ![](http://img.javastack.cn/20210224152923.png) JDK 1.3 就內建了 java.util.Timer 類,可以用來排程 java.util.TimerTask 任務。 ![](http://img.javastack.cn/20210224154501.png) 幾個重要的方法: - schedule:開始排程任務,提供了幾個包裝方法; - cancle:終止任務排程,取消當前排程的所有任務,正在執行的任務不受影響; - purge:從任務佇列中移除所有已取消的任務; 另外,java.util.TimerTask 就是實現了 Runnable 介面,具體任務邏輯則是在 run 方法裡去實現。 ![](http://img.javastack.cn/20210224153049.png) **實現程式碼如下:** ``` /** * timer定時任務 * 來源公眾號:Java技術棧 */ private static void timerTask() throws InterruptedException { Timer timer = new Timer(); TimerTask timerTask = new TimerTask() { @Override public void run() { System.out.println("hi, 歡迎關注:Java技術棧"); } }; // 第一次任務延遲時間 long delay = 2000; // 任務執行頻率 long period = 3 * 1000; // 開始排程 timer.schedule(timerTask, delay, period); // 指定首次執行時間 // timer.schedule(timerTask, DateUtils.addSeconds(new Date(), 5), period); Thread.sleep(20000); // 終止並移除任務 timer.cancel(); timer.purge(); } ``` 這種實現方式比較簡單,可以指定首次執行的延遲時間、首次執行的具體日期時間,以及執行頻率,能滿足日常需要。 另外,需要注意的是,Timer 是執行緒安全的,因為背後是單執行緒在執行所有任務。 **Timer 也會有一些缺陷:** - Timer 是單執行緒的,假如有任務 A,B,C,任務 A 如果執行時間比較長,那麼就會影響任務 B,C 的啟動和執行時間,如果 B,C 執行時間也比較長,那就會相互影響; - Timer 不會捕獲異常,如果 A,B,C 任何一個任務在執行過程中發生異常,就會導致 TImer 整個定時任務停止工作; - Timer 是基於絕對時間排程的,而不是基於相對時間,所以它對系統時間的改變非常敏感; 所以,如果在使用 Timer 的過程中要注意這些缺陷,雖然可以用,但不推薦。 ## 3、ScheduledExecutorService 因 Timer 有一些缺陷,所以不太建議使用 Timer,推薦使用 ScheduledExecutorService: ![](http://img.javastack.cn/20210225110114.png) ScheduledExecutorService 即是 Timer 的替代者,JDK 1.5 併發包引入,是基於[執行緒池](https://mp.weixin.qq.com/s/L2KKLlmOKJUQKfLdFa-1FA)設計的定時任務類: > java.util.concurrent.Executors.newScheduledThreadPool ![](http://img.javastack.cn/20210225102119.png) 上了執行緒池,每個排程任務都會分配到執行緒池中的某一個執行緒去執行,任務就是併發排程執行的,任務之間互不影響。 幾個重要的排程方法: ![](http://img.javastack.cn/20210225104805.png) - schedule:只執行一次排程; - scheduleAtFixedRate:按固定頻率排程,如果執行時間過長,下一次排程會延遲,不會同時執行; - scheduleWithFixedDelay:延遲排程,上一次執行完再加上延遲時間後執行; 另外,可以看出,任務是支援 Runnable 和 Callable 排程的。 **實現程式碼如下:** ``` /** * 執行緒池定時任務 * 來源公眾號:Java技術棧 */ public static void poolTask(){ ScheduledExecutorService pool = Executors.newScheduledThreadPool(10); pool.scheduleAtFixedRate(() -> { System.out.println("hi, 歡迎關注:Java技術棧"); }, 2000, 3000, TimeUnit.MILLISECONDS); } ``` 這是一個按固定頻率排程的任務,建立了 10 個核心執行緒數,首次執行延遲 2 秒,後續每 3 秒執行一次。 這種方式簡單、好用,避免了使用 Timer 帶來的各種問題,推薦使用這種實現方式。 ## 總結 好了,本文棧長分享了 3 種 Java 實現定時任務的方式,也相對簡單,但執行頻率時間設定都太簡單,只適合簡單的業務,不適合實際複雜業務的需求,實際業務要考慮分散式、故障轉移恢復等遠要複雜的多。 本文僅給大家一個參考吧,在不用框架的前提下也能實現定時任務,在小而美的場景,還是很香的。 最後,Java 系列教程還會繼續更新,關注Java技術棧公眾號第一時間推送,還可以在公眾號選單中獲取歷史 Java 教程,都是乾貨。 本節教程所有實戰原始碼已上傳到這個倉庫: > https://github.com/javastacks/javastack 最後,覺得我的文章對你用收穫的話,動動小手,給個在看、轉發,原創不易,棧長需要你的鼓勵。 > 版權申明:本文系公眾號 "Java技術棧" 原創,原創實屬不易,轉載、引用本文內容請註明出處,禁止抄襲、洗稿,請自重,尊重他人勞動成果和知識