Java 定時器的多種實(shí)現(xiàn)方式
一、前言
定時器有三種表現(xiàn)形式:
- 按固定周期定時執(zhí)行
- 延遲一定時間后執(zhí)行
- 指定某個時刻執(zhí)行
JDK 提供了三種常用的定時器實(shí)現(xiàn)方式,分別為:
TimerDelayedQueue延遲隊(duì)列ScheduledThreadPoolExecutor
(1)Timer
發(fā)現(xiàn) eureka 中大量使用了 Timer 定時器:
- Timer 屬于 JDK 比較早期版本的實(shí)現(xiàn),它可以實(shí)現(xiàn)固定周期的任務(wù),以及延遲任務(wù)。
- Timer 會起動一個異步線程去執(zhí)行到期的任務(wù),任務(wù)可以只被調(diào)度執(zhí)行一次,也可以周期性反復(fù)執(zhí)行多次。
Timer 是如何使用的,示例代碼如下:
Timer timer = new Timer();
timer.scheduleAtFixedRate(new TimerTask() {
@Override
public void run() {
// 業(yè)務(wù)代碼
}
}, 5000, 5000); // 5s 后調(diào)度一個周期為 5s 的定時任務(wù)
TimerTask是實(shí)現(xiàn)了Runnable接口的抽象類Timer負(fù)責(zé)調(diào)度和執(zhí)行TimerTask
Timer 的內(nèi)部構(gòu)造,如下:
public class Timer {
// 小根堆,run操作 O(1)、新增 O(logn)、cancel O(logn)
private final TaskQueue queue = new TaskQueue();
// 創(chuàng)建另外線程,任務(wù)處理,會輪詢 queue
private final TimerThread thread = new TimerThread(queue);
public Timer(String name) {
thread.setName(name);
thread.start();
}
}
Timer 它是存在不少設(shè)計(jì)缺陷的,所以并不推薦用戶使用:
Timer是單線程模式,如果某個TimerTask執(zhí)行時間很久,會影響其他任務(wù)的調(diào)度。Timer的任務(wù)調(diào)度是基于系統(tǒng)絕對時間的,如果系統(tǒng)時間不正確,可能會出現(xiàn)問題。TimerTask如果執(zhí)行出現(xiàn)異常,Timer并不會捕獲,會導(dǎo)致線程終止,其他任務(wù)永遠(yuǎn)不會執(zhí)行。
(2)DelayedQueue 延遲隊(duì)列
特征如下:
DelayedQueue是 JDK 中一種可以延遲獲取對象的阻塞隊(duì)列,其內(nèi)部是采用優(yōu)先級隊(duì)列PriorityQueue存儲對象DelayQueue中的每個對象都必須實(shí)現(xiàn)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;
}
@Override
public int compareTo(Delayed o) {
return Long.compare(this.getDelay(TimeUnit.MILLISECONDS), o.getDelay(TimeUnit.MILLISECONDS));
}
@Override
public 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 延遲后開始執(zhí)行任務(wù),每 2s 重復(fù)執(zhí)行一次
}
}
ScheduledThreadPoolExecutor 使用了阻塞隊(duì)列 DelayedWorkQueue。
(4)ScheduledThreadPoolExecutor
線程應(yīng)該是最常見的實(shí)現(xiàn)方案,創(chuàng)建一個線程執(zhí)行任務(wù)即可,舉例幾個不同的寫法,代碼如下
4.1.使用thread + runnable
package com.yezi_tool.demo_basic.test;
import org.springframework.stereotype.Component;
import java.util.Date;
@Component
public 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;
@Component
public 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 定時器的多種實(shí)現(xiàn)方式的詳細(xì)內(nèi)容,更多關(guān)于Java 定時器的實(shí)現(xiàn)的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
MyBatis后端對數(shù)據(jù)庫進(jìn)行增刪改查等操作實(shí)例
Mybatis是appach下開源的一款持久層框架,通過xml與java文件的緊密配合,避免了JDBC所帶來的一系列問題,下面這篇文章主要給大家介紹了關(guān)于MyBatis后端對數(shù)據(jù)庫進(jìn)行增刪改查等操作的相關(guān)資料,需要的朋友可以參考下2022-08-08
淺談BeanPostProcessor加載次序及其對Bean造成的影響分析
這篇文章主要介紹了淺談BeanPostProcessor加載次序及其對Bean造成的影響分析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-04-04
Java狀態(tài)設(shè)計(jì)模式實(shí)現(xiàn)對象狀態(tài)轉(zhuǎn)換的優(yōu)雅方式
Java狀態(tài)設(shè)計(jì)模式通過將對象的行為和狀態(tài)分離,使對象能夠根據(jù)不同的狀態(tài)進(jìn)行不同的行為操作。它通過將狀態(tài)抽象成一個獨(dú)立的類來實(shí)現(xiàn)對狀態(tài)的封裝,從而簡化了復(fù)雜的條件判斷和狀態(tài)轉(zhuǎn)換2023-04-04
Java函數(shù)式編程(一):你好,Lambda表達(dá)式
這篇文章主要介紹了Java函數(shù)式編程(一):你好,Lambda表達(dá)式,本文講解了新老函數(shù)式編程的一些變化,需要的朋友可以參考下2014-09-09
Dapr在Java中的服務(wù)調(diào)用實(shí)戰(zhàn)過程詳解
這篇文章主要為大家介紹了Dapr在Java中的服務(wù)調(diào)用實(shí)戰(zhàn)過程詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-06-06
SpringBoot中如何統(tǒng)一接口返回與全局異常處理詳解
全局異常處理是個比較重要的功能,一般在項(xiàng)目里都會用到,這篇文章主要給大家介紹了關(guān)于SpringBoot中如何統(tǒng)一接口返回與全局異常處理的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考下2021-09-09
Java Iterator接口實(shí)現(xiàn)代碼解析
這篇文章主要介紹了Java Iterator接口實(shí)現(xiàn)代碼解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-05-05

