Java 定時器的多種實現方式
定時器有三種表現形式:
按固定周期定時執行 延遲一定時間后執行 指定某個時刻執行JDK 提供了三種常用的定時器實現方式,分別為:
Timer DelayedQueue 延遲隊列 ScheduledThreadPoolExecutor(1)Timer發現 eureka 中大量使用了 Timer 定時器:
Timer 屬于 JDK 比較早期版本的實現,它可以實現固定周期的任務,以及延遲任務。 Timer 會起動一個異步線程去執行到期的任務,任務可以只被調度執行一次,也可以周期性反復執行多次。Timer 是如何使用的,示例代碼如下:
Timer timer = new Timer();timer.scheduleAtFixedRate(new TimerTask() { @Override public void run() {// 業務代碼 }}, 5000, 5000); // 5s 后調度一個周期為 5s 的定時任務 TimerTask 是實現了 Runnable 接口的抽象類 Timer 負責調度和執行 TimerTask
Timer 的內部構造,如下:
public class Timer { // 小根堆,run操作 O(1)、新增 O(logn)、cancel O(logn) private final TaskQueue queue = new TaskQueue(); // 創建另外線程,任務處理,會輪詢 queue private final TimerThread thread = new TimerThread(queue); public Timer(String name) {thread.setName(name);thread.start(); }}
Timer 它是存在不少設計缺陷的,所以并不推薦用戶使用:
Timer 是單線程模式,如果某個 TimerTask 執行時間很久,會影響其他任務的調度。 Timer 的任務調度是基于系統絕對時間的,如果系統時間不正確,可能會出現問題。 TimerTask 如果執行出現異常,Timer 并不會捕獲,會導致線程終止,其他任務永遠不會執行。(2)DelayedQueue 延遲隊列特征如下:
DelayedQueue 是 JDK 中一種可以延遲獲取對象的阻塞隊列,其內部是采用優先級隊列 PriorityQueue 存儲對象 DelayQueue 中的每個對象都必須實現 Delayed 接口,并重寫 compareTo 和 getDelay 方法DelayedQueue 的使用方法如下:
public class DelayQueueTest { public static void main(String[] args) throws Exception {BlockingQueue<SampleTask> delayQueue = new DelayQueue<>();long now = System.currentTimeMillis();delayQueue.put(new SampleTask(now + 1000));delayQueue.put(new SampleTask(now + 2000));delayQueue.put(new SampleTask(now + 3000));for (int i = 0; i < 3; i++) { System.out.println(new Date(delayQueue.take().getTime()));} } static class SampleTask implements Delayed {long time;public SampleTask(long time) { this.time = time;}public long getTime() { return time;}@Overridepublic int compareTo(Delayed o) { return Long.compare(this.getDelay(TimeUnit.MILLISECONDS), o.getDelay(TimeUnit.MILLISECONDS));}@Overridepublic long getDelay(TimeUnit unit) { return unit.convert(time - System.currentTimeMillis(), TimeUnit.MILLISECONDS);} }}(3)ScheduledThreadPoolExecutor
JDK 提供了功能更加豐富的 ScheduledThreadPoolExecutor
public class ScheduledExecutorServiceTest { public static void main(String[] args) {ScheduledExecutorService executor = Executors.newScheduledThreadPool(5);executor.scheduleAtFixedRate(() -> System.out.println('Hello World'), 1000, 2000, TimeUnit.MILLISECONDS); // 1s 延遲后開始執行任務,每 2s 重復執行一次 }}
ScheduledThreadPoolExecutor 使用了阻塞隊列 DelayedWorkQueue。
(4)ScheduledThreadPoolExecutor線程應該是最常見的實現方案,創建一個線程執行任務即可,舉例幾個不同的寫法,代碼如下
4.1.使用thread + runnable
package com.yezi_tool.demo_basic.test;import org.springframework.stereotype.Component;import java.util.Date;@Componentpublic class ThreadTest { private Integer count = 0; public ThreadTest() {test1(); } public void test1() {new Thread(() -> { while (count < 10) {System.out.println(new Date().toString() + ': ' + count);count++;try { Thread.sleep(3000);} catch (InterruptedException e) { e.printStackTrace();} }}).start(); }}
4.2.使用線程池 + runnable
package com.yezi_tool.demo_basic.test;import org.springframework.stereotype.Component;import java.util.Date;import java.util.concurrent.ExecutorService;import java.util.concurrent.Executors;@Componentpublic class ThreadTest { private static final ExecutorService threadPool = Executors.newFixedThreadPool(5);// 線程池 private Integer count = 0; public ThreadTest() {test2(); } public void test2() {threadPool.execute(() -> { while (count < 10) {System.out.println(new Date().toString() + ': ' + count);count++;try { Thread.sleep(3000);} catch (InterruptedException e) { e.printStackTrace();} }}); }}
以上就是Java 定時器的多種實現方式的詳細內容,更多關于Java 定時器的實現的資料請關注好吧啦網其它相關文章!
相關文章: