java 定時器線程池(ScheduledThreadPoolExecutor)的實(shí)現(xiàn)
前言
定時器線程池提供了定時執(zhí)行任務(wù)的能力,即可以延遲執(zhí)行,可以周期性執(zhí)行。但定時器線程池也還是線程池,最底層實(shí)現(xiàn)還是ThreadPoolExecutor,可以參考我的另外一篇文章多線程–精通ThreadPoolExecutor。
特點(diǎn)說明
1.構(gòu)造函數(shù)
public ScheduledThreadPoolExecutor(int corePoolSize) {
// 對于其他幾個參數(shù)在ThreadPoolExecutor中都已經(jīng)詳細(xì)分析過了,所以這里,將不再展開
// 這里我們可以看到調(diào)用基類中的方法時有個特殊的入?yún)elayedWorkQueue。
// 同時我們也可以發(fā)現(xiàn)這里并沒有設(shè)置延遲時間、周期等參數(shù)入口。
// 所以定時執(zhí)行的實(shí)現(xiàn)必然在DelayedWorkQueue這個對象中了。
super(corePoolSize, Integer.MAX_VALUE, 0, NANOSECONDS,
new DelayedWorkQueue());
}
2.DelayedWorkQueue
DelayedWorkQueue是在ScheduledThreadPoolExecutor的一個內(nèi)部類,實(shí)現(xiàn)了BlockingQueue接口
里面存放任務(wù)隊(duì)列的數(shù)組如下:
private RunnableScheduledFuture<?>[] queue = new RunnableScheduledFuture<?>[INITIAL_CAPACITY];
我們分析過ThreadPoolExecutor,它從任務(wù)隊(duì)列中獲取任務(wù)的方式為poll和take兩種,所以看一下poll和take兩個方法的源碼,回顧一下,ThreadPoolExecutor它會調(diào)用poll或take方法,先poll,再take,只要其中一個接口有返回就行
public RunnableScheduledFuture<?> poll() {
final ReentrantLock lock = this.lock;
lock.lock();
try {
RunnableScheduledFuture<?> first = queue[0];
// 這里有個getDelay,這是關(guān)鍵點(diǎn),獲取執(zhí)行延時時間
// 但是如果我們有延時設(shè)置的話,這就返回空了,然后就會調(diào)用take方法
if (first == null || first.getDelay(NANOSECONDS) > 0)
return null;
else
return finishPoll(first);
} finally {
lock.unlock();
}
}
public RunnableScheduledFuture<?> take() throws InterruptedException {
final ReentrantLock lock = this.lock;
lock.lockInterruptibly();
try {
for (;;) {
RunnableScheduledFuture<?> first = queue[0];
if (first == null)
available.await();
else {
// 獲取延時時間
long delay = first.getDelay(NANOSECONDS);
if (delay <= 0)
return finishPoll(first);
first = null; // don't retain ref while waiting
if (leader != null)
available.await();
else {
Thread thisThread = Thread.currentThread();
leader = thisThread;
try {
// 使用鎖,執(zhí)行延時等待。
// 使用鎖,執(zhí)行延時等待。
// 使用鎖,執(zhí)行延時等待。
available.awaitNanos(delay);
} finally {
if (leader == thisThread)
leader = null;
}
}
}
}
} finally {
if (leader == null && queue[0] != null)
available.signal();
lock.unlock();
}
}
3.RunnableScheduledFuture
在ScheduledThreadPoolExecutor內(nèi)部有一個ScheduledFutureTask類實(shí)現(xiàn)了RunnableScheduledFuture,ScheduledFutureTask這個類采用了裝飾者設(shè)計(jì)模式,在執(zhí)行Runnable的方法基礎(chǔ)上還執(zhí)行了一些額外的功能。
我們需要特別注意幾個參數(shù)period、time。
(1)time
首先看一下time的作用,可以發(fā)現(xiàn)time是用于獲取執(zhí)行延時時間的,也就是delay是根據(jù)time生成的
public long getDelay(TimeUnit unit) {
return unit.convert(time - now(), NANOSECONDS);
}
(2)period
這個參數(shù)不是說設(shè)置執(zhí)行幾個周期,而是用于判斷是否需要按周期執(zhí)行,以及執(zhí)行周期,也就是本次執(zhí)行與下次執(zhí)行間隔的時間
// 判斷是否需要按周期執(zhí)行,如果周期設(shè)置成0,不是無間隔執(zhí)行,而是只執(zhí)行一次,這個需要特別注意
public boolean isPeriodic() {
return period != 0;
}
private void setNextRunTime() {
long p = period;
if (p > 0)
// 這里將周期加給time,這樣獲取的延遲時間就是周期時間了。
time += p;
else
time = triggerTime(-p);
}
(3)執(zhí)行
public void run() {
// 先判斷是否為周期性的任務(wù)
boolean periodic = isPeriodic();
if (!canRunInCurrentRunState(periodic))
cancel(false);
else if (!periodic)
// 如果不是周期性的,就執(zhí)行調(diào)用父類的run方法,也就是構(gòu)造函數(shù)中傳入的Runnable對象的run方法。
ScheduledFutureTask.super.run();
// 在if的括號中先執(zhí)行了任務(wù)
else if (ScheduledFutureTask.super.runAndReset()) {
// 如果是周期性的,就需要設(shè)置下次執(zhí)行的時間,然后利用reExecutePeriodic方法,將任務(wù)再次丟入任務(wù)隊(duì)列中。
// 這里尤其需要注意的是if中的邏輯執(zhí)行失敗,如果沒有捕捉異常,那么后面的邏輯就不會再執(zhí)行了,也就是說中間有一次執(zhí)行失敗,后面這個周期性的任務(wù)就失效了。
setNextRunTime();
reExecutePeriodic(outerTask);
}
}
總結(jié)
ScheduledThreadPoolExecutor通過time參數(shù),設(shè)置當(dāng)前任務(wù)執(zhí)行的等待時間,再通過period設(shè)置任務(wù)下次執(zhí)行需要等待的時間。這兩個參數(shù)都不是設(shè)置在線程池中的,而是攜帶在任務(wù)中的,這就可以把線程池和任務(wù)進(jìn)行完全解耦。
注意點(diǎn):
(1)任務(wù)的執(zhí)行等待時間是在隊(duì)列的take方法中的。
(2)period參數(shù)設(shè)置成0,任務(wù)將只會執(zhí)行一次,而不會執(zhí)行多次
(3)如果要自己實(shí)現(xiàn)周期性Task,周期性任務(wù)在執(zhí)行過程中,一定要注意捕捉異常,否則某一次執(zhí)行失敗,將導(dǎo)致后續(xù)的任務(wù)周期失效,任務(wù)將不再繼續(xù)執(zhí)行。
到此這篇關(guān)于java 定時器線程池(ScheduledThreadPoolExecutor)的實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)java 定時器線程池內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- Java自帶定時任務(wù)ScheduledThreadPoolExecutor實(shí)現(xiàn)定時器和延時加載功能
- Java調(diào)度線程池ScheduledThreadPoolExecutor不執(zhí)行問題分析
- java高并發(fā)ScheduledThreadPoolExecutor類深度解析
- 詳解Java ScheduledThreadPoolExecutor的踩坑與解決方法
- java高并發(fā)ScheduledThreadPoolExecutor與Timer區(qū)別
- Java使用quartz實(shí)現(xiàn)定時任務(wù)示例詳解
- Java實(shí)現(xiàn)定時任務(wù)最簡單的3種方法
- Java項(xiàng)目實(shí)現(xiàn)定時任務(wù)的三種方法
- Java定時任務(wù)ScheduledThreadPoolExecutor示例詳解
相關(guān)文章
maven無法依賴spring-cloud-stater-zipkin的解決方案
這篇文章主要介紹了maven無法依賴spring-cloud-stater-zipkin如何解決,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-05-05
Java中綴表達(dá)式轉(zhuǎn)后綴表達(dá)式流程詳解
中綴表達(dá)式是一個通用的算術(shù)或邏輯公式表示方法。,中綴表達(dá)式不容易被計(jì)算機(jī)解析,但仍被許多程序語言使用,因?yàn)樗先藗兊钠毡橛梅?。本文介紹了實(shí)現(xiàn)中綴表達(dá)式的方法,需要的可以參考一下2022-09-09
Java 實(shí)戰(zhàn)項(xiàng)目錘煉之在線購書商城系統(tǒng)的實(shí)現(xiàn)流程
讀萬卷書不如行萬里路,只學(xué)書上的理論是遠(yuǎn)遠(yuǎn)不夠的,只有在實(shí)戰(zhàn)中才能獲得能力的提升,本篇文章手把手帶你用java+jsp+mysql+servlet+ajax實(shí)現(xiàn)一個在線購書商城系統(tǒng),大家可以在過程中查缺補(bǔ)漏,提升水平2021-11-11
Spring整合SpringMVC與Mybatis(SSM)實(shí)現(xiàn)完整登錄功能流程詳解
開學(xué)學(xué)校開始講servlet 后期要求做一個登錄功能,這個使用SSM先只做個簡單的只帶登錄功能的,且項(xiàng)目使用了MyBatis-Plus來簡化開發(fā)流程??辞闆r決定要不要升級功能或者換個寫法2022-09-09
EDI中JAVA通過FTP工具實(shí)現(xiàn)文件上傳下載實(shí)例
這篇文章主要介紹了EDI中JAVA通過FTP工具實(shí)現(xiàn)文件上傳下載實(shí)例,具有一定的參考價(jià)值,有需要的可以了解一下。2016-11-11
Spring5學(xué)習(xí)之基礎(chǔ)知識總結(jié)
這篇文章主要介紹了Spring5學(xué)習(xí)之基礎(chǔ)知識總結(jié),文中有非常詳細(xì)的代碼示例,對正在學(xué)習(xí)java的小伙伴們有非常好的幫助,需要的朋友可以參考下2021-05-05
Nacos配置文件使用經(jīng)驗(yàn)及CAP原則詳解
這篇文章主要為大家介紹了Nacos配置文件使用經(jīng)驗(yàn)及CAP規(guī)則詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2024-02-02

