Java 實現(xiàn)定時任務的三種方法
是的,不用任何框架,用我們樸素的 Java 編程語言就能實現(xiàn)定時任務。
今天,棧長就介紹 3 種實現(xiàn)方法,教你如何使用 JDK 實現(xiàn)定時任務!
1、 sleep這也是我們最常用的 sleep 休眠大法,不只是當作休眠用,我們還可以利用它很輕松的能實現(xiàn)一個簡單的定時任務。
實現(xiàn)邏輯:
新開一個線程,添加一個 for/ while 死循環(huán),然后在死循環(huán)里面添加一個 sleep 休眠邏輯,讓程序每隔 N 秒休眠再執(zhí)行一次,這樣就達到了一個簡單定時任務的效果。
實現(xiàn)代碼如下:
private static void sleepTask() { new Thread(() -> { while (true) { System.out.println('hi, 歡迎關注:Java技術棧'); try { // 每隔3秒執(zhí)行一次 Thread.sleep(3000); } catch (InterruptedException e) { e.printStackTrace(); } } }).start();}
這種方式比較傻瓜化了,只能按固定頻率運行,不能指定具體運行的時間。
另外,上面的箭頭語法,棧長使用了 JDK 8 中的 Lambda 表達式,這里就不再撰述了,Java 8 系列實戰(zhàn)教程我都寫了一堆了,不清楚的可以關注公眾號:Java技術棧,在后臺回復 'java' 閱讀,我都整理好了。
2、Timer來看下 JDK 自帶的 java.util.Timer 類:
JDK 1.3 就內(nèi)置了 java.util.Timer 類,可以用來調(diào)度 java.util.TimerTask 任務。
幾個重要的方法:
schedule:開始調(diào)度任務,提供了幾個包裝方法; cancle:終止任務調(diào)度,取消當前調(diào)度的所有任務,正在運行的任務不受影響; purge:從任務隊列中移除所有已取消的任務;另外,java.util.TimerTask 就是實現(xiàn)了 Runnable 接口,具體任務邏輯則是在 run 方法里去實現(xiàn)。
實現(xiàn)代碼如下:
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; // 任務執(zhí)行頻率 long period = 3 * 1000; // 開始調(diào)度 timer.schedule(timerTask, delay, period); // 指定首次運行時間// timer.schedule(timerTask, DateUtils.addSeconds(new Date(), 5), period); Thread.sleep(20000); // 終止并移除任務 timer.cancel(); timer.purge();}
這種實現(xiàn)方式比較簡單,可以指定首次執(zhí)行的延遲時間、首次執(zhí)行的具體日期時間,以及執(zhí)行頻率,能滿足日常需要。
另外,需要注意的是,Timer 是線程安全的,因為背后是單線程在執(zhí)行所有任務。
Timer 也會有一些缺陷:
Timer 是單線程的,假如有任務 A,B,C,任務 A 如果執(zhí)行時間比較長,那么就會影響任務 B,C 的啟動和執(zhí)行時間,如果 B,C 執(zhí)行時間也比較長,那就會相互影響; Timer 不會捕獲異常,如果 A,B,C 任何一個任務在執(zhí)行過程中發(fā)生異常,就會導致 TImer 整個定時任務停止工作; Timer 是基于絕對時間調(diào)度的,而不是基于相對時間,所以它對系統(tǒng)時間的改變非常敏感;所以,如果在使用 Timer 的過程中要注意這些缺陷,雖然可以用,但不推薦。
3、ScheduledExecutorService因 Timer 有一些缺陷,所以不太建議使用 Timer,推薦使用 ScheduledExecutorService:
ScheduledExecutorService 即是 Timer 的替代者,JDK 1.5 并發(fā)包引入,是基于線程池設計的定時任務類:
java.util.concurrent.Executors.newScheduledThreadPool
上了線程池,每個調(diào)度任務都會分配到線程池中的某一個線程去執(zhí)行,任務就是并發(fā)調(diào)度執(zhí)行的,任務之間互不影響。
幾個重要的調(diào)度方法:
另外,可以看出,任務是支持 Runnable 和 Callable 調(diào)度的。
實現(xiàn)代碼如下:
public static void poolTask(){ ScheduledExecutorService pool = Executors.newScheduledThreadPool(10); pool.scheduleAtFixedRate(() -> { System.out.println('hi, 歡迎關注:Java技術棧'); }, 2000, 3000, TimeUnit.MILLISECONDS);}
這是一個按固定頻率調(diào)度的任務,創(chuàng)建了 10 個核心線程數(shù),首次執(zhí)行延遲 2 秒,后續(xù)每 3 秒執(zhí)行一次。
這種方式簡單、好用,避免了使用 Timer 帶來的各種問題,推薦使用這種實現(xiàn)方式。
總結好了,本文棧長分享了 3 種 Java 實現(xiàn)定時任務的方式,也相對簡單,但執(zhí)行頻率時間設置都太簡單,只適合簡單的業(yè)務,不適合實際復雜業(yè)務的需求,實際業(yè)務要考慮分布式、故障轉(zhuǎn)移恢復等遠要復雜的多。
本文僅給大家一個參考吧,在不用框架的前提下也能實現(xiàn)定時任務,在小而美的場景,還是很香的。
最后,Java 系列教程還會繼續(xù)更新,關注Java技術棧公眾號第一時間推送,還可以在公眾號菜單中獲取歷史 Java 教程,都是干貨。
本節(jié)教程所有實戰(zhàn)源碼已上傳到這個倉庫:
https://github.com/javastacks/javastack
最后,覺得我的文章對你用收獲的話,動動小手,給個在看、轉(zhuǎn)發(fā),原創(chuàng)不易,棧長需要你的鼓勵。
以上就是Java 實現(xiàn)定時任務的三種方法的詳細內(nèi)容,更多關于Java 實現(xiàn)定時任務的資料請關注好吧啦網(wǎng)其它相關文章!
相關文章:
1. Java 3D的動畫展示(Part1-使用JMF)2. 解決docker與vmware的沖突問題3. IntelliJ Idea 2020.1 正式發(fā)布,官方支持中文(必看)4. Django ORM實現(xiàn)按天獲取數(shù)據(jù)去重求和例子5. Python基于jieba, wordcloud庫生成中文詞云6. Django中的AutoField字段使用7. IntelliJ IDEA設置自動提示功能快捷鍵的方法8. asp.net core應用docke部署到centos7的全過程9. java使用FFmpeg合成視頻和音頻并獲取視頻中的音頻等操作(實例代碼詳解)10. PHP設計模式之 策略模式Strategy詳解【對象行為型】
