兩階段終止模式:如何優(yōu)雅地終止線程网杆?

如何理解兩階段終止模式

Java 語言的 Thread 類中曾經(jīng)提供了一個 stop() 方法,用來終止線程也颤,現(xiàn)在已不建議使用洋幻,原因是這個方法要求線程立即終止,被終止的線程沒有機會料理后事翅娶。既然不建議使用 stop() 方法文留,又該如何優(yōu)雅地終止線程呢?
Java有一套成熟的方案竭沫,叫兩階段終止模式燥翅。就是將終止過程分成兩個階段,其中第一個階段主要是線程 T1 向線程 T2發(fā)送終止指令输吏,而第二階段則是線程 T2響應終止指令权旷。


兩階段終止模式示意圖

Java語言的終止指令是什么

Java 中的線程狀態(tài)轉換圖

從這個圖里你會發(fā)現(xiàn)替蛉,Java 線程進入終止狀態(tài)的前提是線程進入 RUNNABLE 狀態(tài)贯溅,而實際上線程也可能處在休眠狀態(tài),也就是說躲查,我們要想終止一個線程它浅,首先要把線程的狀態(tài)從休眠狀態(tài)轉換到 RUNNABLE 狀態(tài)。如何做到呢镣煮?這個要靠 Java Thread 類提供的interrupt() 方法姐霍,它可以將休眠狀態(tài)的線程轉換到 RUNNABLE 狀態(tài)

線程轉換到 RUNNABLE 狀態(tài)之后典唇,我們?nèi)绾卧賹⑵浣K止呢镊折?RUNNABLE 狀態(tài)轉換到終止狀態(tài),優(yōu)雅的方式是讓 Java 線程自己執(zhí)行完 run() 方法介衔,所以一般我們采用的方法是設置一個標志位恨胚,然后線程會在合適的時機檢查這個標志位,如果發(fā)現(xiàn)符合終止條件炎咖,則自動退出 run() 方法赃泡。這個過程其實就是我們前面提到的第二階段:響應終止指令寒波。

綜合上面這兩點,我們能總結出終止指令升熊,其實包括兩方面內(nèi)容:interrupt() 方法和線程終止的標志位俄烁。

用兩階段終止模式終止監(jiān)控操作

實際工作中,有些監(jiān)控系統(tǒng)需要動態(tài)地采集一些數(shù)據(jù)级野,一般都是監(jiān)控系統(tǒng)發(fā)送采集指令給被監(jiān)控系統(tǒng)的監(jiān)控代理页屠,監(jiān)控代理接收到指令之后,從監(jiān)控目標收集數(shù)據(jù)蓖柔,然后回傳給監(jiān)控系統(tǒng)卷中,詳細過程如下圖所示。出于對性能的考慮渊抽,動態(tài)采集功能一般都會有終止操作蟆豫。


動態(tài)采集功能示意圖

下面的示例代碼是監(jiān)控代理簡化之后的實現(xiàn):

class Proxy {
  // 設置自己的線程終止標志位。在線程的 run() 方法中如果調用第三方類庫提供的方法懒闷,我們沒有辦法保證第三方類庫正確處理了線程的中斷異常十减,例如第三方類庫在捕獲到 Thread.sleep() 方法拋出的中斷異常后,沒有重新設置線程的中斷狀態(tài)愤估,那么就會導致線程不能夠正常終止帮辟。
  volatile boolean terminated = false;
  boolean started = false;
  // 采集線程
  Thread rptThread;
  // 啟動采集功能
  synchronized void start(){
    // 不允許同時啟動多個采集線程
    if (started) {
      return;
    }
    started = true;
    terminated = false;
    rptThread = new Thread(()->{
      while (!terminated){
        // 省略采集、回傳實現(xiàn)
        report();
        // 每隔兩秒鐘采集玩焰、回傳一次數(shù)據(jù)
        try {
          Thread.sleep(2000);
        } catch (InterruptedException e){
          // 重新設置線程中斷狀態(tài)
          Thread.currentThread().interrupt();
        }
      }
      // 執(zhí)行到此處說明線程馬上終止
      started = false;
    });
    rptThread.start();
  }
  // 終止采集功能
  synchronized void stop(){
    // 設置中斷標志位
    terminated = true;
    // 中斷線程 rptThread
    rptThread.interrupt();
  }
}

如何優(yōu)雅地終止線程池

線程池提供了兩個方法:shutdown()和shutdownNow()由驹。這兩個方法有什么區(qū)別呢?
shutdown():線程池執(zhí)行 shutdown() 后昔园,會拒絕接收新的任務蔓榄,但是會等待線程池中正在執(zhí)行的任務和已經(jīng)進入阻塞隊列的任務都執(zhí)行完之后才最終關閉線程池。

shutdownNow() :線程池執(zhí)行 shutdownNow() 后默刚,會拒絕接收新的任務甥郑,同時還會中斷線程池中正在執(zhí)行的任務,已經(jīng)進入阻塞隊列的任務也被剝奪了執(zhí)行的機會荤西,不過這些被剝奪執(zhí)行機會的任務會作為 shutdownNow() 方法的返回值返回澜搅。因為 shutdownNow() 方法會中斷正在執(zhí)行的線程,所以提交到線程池的任務邪锌,如果需要優(yōu)雅地結束勉躺,就需要正確地處理線程中斷。

shutdown() 和 shutdownNow() 方法實質上使用的也是兩階段終止模式觅丰,只是終止指令的范圍不同而已饵溅,前者只影響阻塞隊列接收任務,后者范圍擴大到線程池中所有的任務舶胀。

?著作權歸作者所有,轉載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末概说,一起剝皮案震驚了整個濱河市碧注,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌糖赔,老刑警劉巖萍丐,帶你破解...
    沈念sama閱讀 221,548評論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異放典,居然都是意外死亡逝变,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,497評論 3 399
  • 文/潘曉璐 我一進店門奋构,熙熙樓的掌柜王于貴愁眉苦臉地迎上來壳影,“玉大人,你說我怎么就攤上這事弥臼⊙邕郑” “怎么了?”我有些...
    開封第一講書人閱讀 167,990評論 0 360
  • 文/不壞的土叔 我叫張陵径缅,是天一觀的道長掺栅。 經(jīng)常有香客問我,道長纳猪,這世上最難降的妖魔是什么氧卧? 我笑而不...
    開封第一講書人閱讀 59,618評論 1 296
  • 正文 為了忘掉前任,我火速辦了婚禮氏堤,結果婚禮上沙绝,老公的妹妹穿的比我還像新娘。我一直安慰自己鼠锈,他們只是感情好闪檬,可當我...
    茶點故事閱讀 68,618評論 6 397
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著脚祟,像睡著了一般谬以。 火紅的嫁衣襯著肌膚如雪强饮。 梳的紋絲不亂的頭發(fā)上由桌,一...
    開封第一講書人閱讀 52,246評論 1 308
  • 那天,我揣著相機與錄音邮丰,去河邊找鬼行您。 笑死,一個胖子當著我的面吹牛剪廉,可吹牛的內(nèi)容都是我干的娃循。 我是一名探鬼主播,決...
    沈念sama閱讀 40,819評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼斗蒋,長吁一口氣:“原來是場噩夢啊……” “哼捌斧!你這毒婦竟也來了笛质?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 39,725評論 0 276
  • 序言:老撾萬榮一對情侶失蹤捞蚂,失蹤者是張志新(化名)和其女友劉穎妇押,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體姓迅,經(jīng)...
    沈念sama閱讀 46,268評論 1 320
  • 正文 獨居荒郊野嶺守林人離奇死亡敲霍,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,356評論 3 340
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了丁存。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片肩杈。...
    茶點故事閱讀 40,488評論 1 352
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖解寝,靈堂內(nèi)的尸體忽然破棺而出扩然,到底是詐尸還是另有隱情,我是刑警寧澤聋伦,帶...
    沈念sama閱讀 36,181評論 5 350
  • 正文 年R本政府宣布与学,位于F島的核電站,受9級特大地震影響嘉抓,放射性物質發(fā)生泄漏索守。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,862評論 3 333
  • 文/蒙蒙 一抑片、第九天 我趴在偏房一處隱蔽的房頂上張望卵佛。 院中可真熱鬧,春花似錦敞斋、人聲如沸截汪。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,331評論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽衙解。三九已至,卻和暖如春焰枢,著一層夾襖步出監(jiān)牢的瞬間蚓峦,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,445評論 1 272
  • 我被黑心中介騙來泰國打工济锄, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留暑椰,地道東北人。 一個月前我還...
    沈念sama閱讀 48,897評論 3 376
  • 正文 我出身青樓荐绝,卻偏偏與公主長得像一汽,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子低滩,可洞房花燭夜當晚...
    茶點故事閱讀 45,500評論 2 359

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