定時(shí)器線程池(ScheduledThreadPoolExecutor)

前言

定時(shí)器線程池提供了定時(shí)執(zhí)行任務(wù)的能力伟叛,即可以延遲執(zhí)行私痹,可以周期性執(zhí)行。但定時(shí)器線程池也還是線程池统刮,最底層實(shí)現(xiàn)還是ThreadPoolExecutor紊遵,可以參考我的另外一篇文章多線程--精通ThreadPoolExecutor

特點(diǎn)說(shuō)明

1.構(gòu)造函數(shù)

 public ScheduledThreadPoolExecutor(int corePoolSize) {
 // 對(duì)于其他幾個(gè)參數(shù)在ThreadPoolExecutor中都已經(jīng)詳細(xì)分析過(guò)了侥蒙,所以這里,將不再展開
 // 這里我們可以看到調(diào)用基類中的方法時(shí)有個(gè)特殊的入?yún)elayedWorkQueue暗膜。
 // 同時(shí)我們也可以發(fā)現(xiàn)這里并沒(méi)有設(shè)置延遲時(shí)間、周期等參數(shù)入口鞭衩。
 // 所以定時(shí)執(zhí)行的實(shí)現(xiàn)必然在DelayedWorkQueue這個(gè)對(duì)象中了学搜。
        super(corePoolSize, Integer.MAX_VALUE, 0, NANOSECONDS,
              new DelayedWorkQueue());
    }

2.DelayedWorkQueue

DelayedWorkQueue是在ScheduledThreadPoolExecutor的一個(gè)內(nèi)部類,實(shí)現(xiàn)了BlockingQueue接口
里面存放任務(wù)隊(duì)列的數(shù)組如下:

private RunnableScheduledFuture<?>[] queue =
            new RunnableScheduledFuture<?>[INITIAL_CAPACITY];

我們分析過(guò)ThreadPoolExecutor论衍,它從任務(wù)隊(duì)列中獲取任務(wù)的方式為poll和take兩種瑞佩,所以看一下poll和take兩個(gè)方法的源碼,回顧一下坯台,ThreadPoolExecutor它會(huì)調(diào)用poll或take方法炬丸,先poll,再take蜒蕾,只要其中一個(gè)接口有返回就行

public RunnableScheduledFuture<?> poll() {
            final ReentrantLock lock = this.lock;
            lock.lock();
            try {
                RunnableScheduledFuture<?> first = queue[0];
                // 這里有個(gè)getDelay稠炬,這是關(guān)鍵點(diǎn),獲取執(zhí)行延時(shí)時(shí)間
                // 但是如果我們有延時(shí)設(shè)置的話滥搭,這就返回空了酸纲,然后就會(huì)調(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 {
                    // 獲取延時(shí)時(shí)間
                        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í)行延時(shí)等待瑟匆。
                            // 使用鎖闽坡,執(zhí)行延時(shí)等待。
                            // 使用鎖愁溜,執(zhí)行延時(shí)等待疾嗅。
                                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)部有一個(gè)ScheduledFutureTask類實(shí)現(xiàn)了RunnableScheduledFuture,ScheduledFutureTask這個(gè)類采用了裝飾者設(shè)計(jì)模式冕象,在執(zhí)行Runnable的方法基礎(chǔ)上還執(zhí)行了一些額外的功能代承。
我們需要特別注意幾個(gè)參數(shù)period、time渐扮。
(1)time
首先看一下time的作用论悴,可以發(fā)現(xiàn)time是用于獲取執(zhí)行延時(shí)時(shí)間的掖棉,也就是delay是根據(jù)time生成的

public long getDelay(TimeUnit unit) {
            return unit.convert(time - now(), NANOSECONDS);
        }
```
(2)period
這個(gè)參數(shù)不是說(shuō)設(shè)置執(zhí)行幾個(gè)周期,而是用于判斷是否需要按周期執(zhí)行膀估,以及執(zhí)行周期幔亥,也就是本次執(zhí)行與下次執(zhí)行間隔的時(shí)間
``` java
// 判斷是否需要按周期執(zhí)行,如果周期設(shè)置成0察纯,不是無(wú)間隔執(zhí)行帕棉,而是只執(zhí)行一次,這個(gè)需要特別注意
  public boolean isPeriodic() {
            return period != 0;
        }
```
``` java
 private void setNextRunTime() {
            long p = period;
            if (p > 0)
            // 這里將周期加給time饼记,這樣獲取的延遲時(shí)間就是周期時(shí)間了香伴。
                time += p;
            else
                time = triggerTime(-p);
        }
```
(3)執(zhí)行
``` java
 public void run() {
            // 先判斷是否為周期性的任務(wù)
            boolean periodic = isPeriodic();
            if (!canRunInCurrentRunState(periodic))
                cancel(false);
            else if (!periodic)
            // 如果不是周期性的,就執(zhí)行調(diào)用父類的run方法具则,也就是構(gòu)造函數(shù)中傳入的Runnable對(duì)象的run方法即纲。
                ScheduledFutureTask.super.run();
                // 在if的括號(hào)中先執(zhí)行了任務(wù)
            else if (ScheduledFutureTask.super.runAndReset()) {
            // 如果是周期性的,就需要設(shè)置下次執(zhí)行的時(shí)間乡洼,然后利用reExecutePeriodic方法崇裁,將任務(wù)再次丟入任務(wù)隊(duì)列中。
            // 這里尤其需要注意的是if中的邏輯執(zhí)行失敗束昵,如果沒(méi)有捕捉異常拔稳,那么后面的邏輯就不會(huì)再執(zhí)行了,也就是說(shuō)中間有一次執(zhí)行失敗锹雏,后面這個(gè)周期性的任務(wù)就失效了巴比。
                setNextRunTime();
                reExecutePeriodic(outerTask);
            }
        }
```
# 總結(jié)
ScheduledThreadPoolExecutor通過(guò)time參數(shù),設(shè)置當(dāng)前任務(wù)執(zhí)行的等待時(shí)間礁遵,再通過(guò)period設(shè)置任務(wù)下次執(zhí)行需要等待的時(shí)間轻绞。這兩個(gè)參數(shù)都不是設(shè)置在線程池中的,而是攜帶在任務(wù)中的佣耐,這就可以把線程池和任務(wù)進(jìn)行完全解耦政勃。
注意點(diǎn):
(1)任務(wù)的執(zhí)行等待時(shí)間是在隊(duì)列的take方法中的。
(2)period參數(shù)設(shè)置成0兼砖,任務(wù)將只會(huì)執(zhí)行一次奸远,而不會(huì)執(zhí)行多次
(3)如果要自己實(shí)現(xiàn)周期性Task,周期性任務(wù)在執(zhí)行過(guò)程中讽挟,一定要注意捕捉異常懒叛,否則某一次執(zhí)行失敗,將導(dǎo)致后續(xù)的任務(wù)周期失效耽梅,任務(wù)將不再繼續(xù)執(zhí)行薛窥。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市眼姐,隨后出現(xiàn)的幾起案子诅迷,更是在濱河造成了極大的恐慌佩番,老刑警劉巖,帶你破解...
    沈念sama閱讀 221,888評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件竟贯,死亡現(xiàn)場(chǎng)離奇詭異答捕,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)屑那,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,677評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)艘款,“玉大人持际,你說(shuō)我怎么就攤上這事』┡兀” “怎么了蜘欲?”我有些...
    開封第一講書人閱讀 168,386評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)晌柬。 經(jīng)常有香客問(wèn)我姥份,道長(zhǎng),這世上最難降的妖魔是什么年碘? 我笑而不...
    開封第一講書人閱讀 59,726評(píng)論 1 297
  • 正文 為了忘掉前任澈歉,我火速辦了婚禮,結(jié)果婚禮上屿衅,老公的妹妹穿的比我還像新娘埃难。我一直安慰自己,他們只是感情好涤久,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,729評(píng)論 6 397
  • 文/花漫 我一把揭開白布涡尘。 她就那樣靜靜地躺著,像睡著了一般响迂。 火紅的嫁衣襯著肌膚如雪考抄。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,337評(píng)論 1 310
  • 那天蔗彤,我揣著相機(jī)與錄音川梅,去河邊找鬼。 笑死幕与,一個(gè)胖子當(dāng)著我的面吹牛挑势,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播啦鸣,決...
    沈念sama閱讀 40,902評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼潮饱,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了诫给?” 一聲冷哼從身側(cè)響起香拉,我...
    開封第一講書人閱讀 39,807評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤啦扬,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后凫碌,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體扑毡,經(jīng)...
    沈念sama閱讀 46,349評(píng)論 1 318
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,439評(píng)論 3 340
  • 正文 我和宋清朗相戀三年盛险,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了瞄摊。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,567評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡苦掘,死狀恐怖换帜,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情鹤啡,我是刑警寧澤惯驼,帶...
    沈念sama閱讀 36,242評(píng)論 5 350
  • 正文 年R本政府宣布,位于F島的核電站递瑰,受9級(jí)特大地震影響祟牲,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜抖部,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,933評(píng)論 3 334
  • 文/蒙蒙 一说贝、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧您朽,春花似錦狂丝、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,420評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至讯屈,卻和暖如春蛋哭,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背涮母。 一陣腳步聲響...
    開封第一講書人閱讀 33,531評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工谆趾, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人叛本。 一個(gè)月前我還...
    沈念sama閱讀 48,995評(píng)論 3 377
  • 正文 我出身青樓沪蓬,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親来候。 傳聞我的和親對(duì)象是個(gè)殘疾皇子跷叉,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,585評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容