Java 停止線程需要注意的地方
Java中停止線程的原則是什么?
在 Java 中, 最好的停止線程的方式是使用中斷 interrupt, 但是這僅僅是會通知到被終止的線程 '你該停止運(yùn)行了', 被終止的線程自身擁有決定權(quán) (決定是否、以及何時(shí)停止), 這依賴于請求停止方和被停止方都遵守一種約定好的編碼規(guī)范.
任務(wù)和線程的啟動很容易. 在大多數(shù)時(shí)候, 我們都會讓它們運(yùn)行直到結(jié)束, 或者讓它們自行停止.然而, 有時(shí)候我們希望提前結(jié)束任務(wù)或線程, 或許是因?yàn)橛脩羧∠瞬僮?或者服務(wù)需要被快速關(guān)閉, 或者是運(yùn)行超時(shí)或出錯(cuò)了. 要使任務(wù)和線程能安全、快速、可靠地停止下來, 并不是一件容易的事. Java沒有提供任何機(jī)制來安全地終止線程. 但它提供了中斷 (Interruption), 這是一種協(xié)作機(jī)制,能夠使一個(gè)線程終止另一個(gè)線程的當(dāng)前工作. 這種協(xié)作式的方法是必要的, 我們很少希望某個(gè)任務(wù)、線程或服務(wù)立即停止,因?yàn)檫@種立即停止會使共享的數(shù)據(jù)結(jié)構(gòu)處于不一致的狀態(tài). 相反, 在編寫任務(wù)和服務(wù)時(shí)可以使用一種協(xié)作的方式: 當(dāng)需要停止時(shí),它們首先會清除當(dāng)前正在執(zhí)行的工作, 然后再結(jié)束. 這提供了更好的靈活性, 因?yàn)槿蝿?wù)本身的代碼比發(fā)出取消請求的代碼更清楚如何執(zhí)行清除工作. 生命周期結(jié)束 (End-of-Lifecycle) 的問題會使任務(wù)、服務(wù)以及程序的設(shè)計(jì)和實(shí)現(xiàn)等過程變得復(fù)雜, 而這個(gè)在程序設(shè)計(jì)中非常重要的要素卻經(jīng)常被忽略. 一個(gè)在行為良好的軟件與勉強(qiáng)運(yùn)的軟件之間的最主要區(qū)別就是, 行為良好的軟件能很完善地處理失敗、關(guān)閉和取消等過程.處理中斷的最好方法是什么?
優(yōu)先選擇在方法上拋出異常.
用 throws InterruptedException 標(biāo)記你的方法, 不采用 try 語句塊捕獲異常,以便于該異常可以傳遞到頂層, 讓run方法可以捕獲這一異常, 例如:
void subTask() throws InterruptedExceptionsleep(delay);}
由于 run 方法內(nèi)無法拋出 checked Exception (只能用 try catch), 頂層方法必須處理該異常, 避免了漏掉或者被吞掉的情況, 增強(qiáng)了代碼的健壯性.
如果不能拋出中斷, 要怎么做?
如果不想或無法傳遞 InterruptedException (例如用 run 方法的時(shí)候, 就不讓該方法 throws InterruptedException), 那么應(yīng)該選擇在 catch 子句中調(diào)用 Thread.currentThread().interrupt() 來恢復(fù)設(shè)置中斷狀態(tài), 以便于在后續(xù)的執(zhí)行依然能夠檢查到剛才發(fā)生了中斷.
代碼演示詳見視頻, 在這里, 線程在sleep期間被中斷, 并且由 catch 捕獲到該中斷, 并重新設(shè)置了中斷狀態(tài), 以便于可以在下一個(gè)循環(huán)的時(shí)候檢測到中斷狀態(tài), 正常退出.
為什么用 volatile 停止線程不夠全面?
解答: 這種做法是錯(cuò)誤的, 或者說是不夠全面的, 在某些情況下雖然可用, 但是某些情況下有嚴(yán)重問題。
這種方法在《Java并發(fā)編程實(shí)戰(zhàn)》中被明確指出了缺陷, 我們一起來看看缺陷在哪里:
此方法錯(cuò)誤的原因在于, 如果我們遇到了線程長時(shí)間阻塞 (這是一種很常見的情況, 例如生產(chǎn)者消費(fèi)者模式中就存在這樣的情況), 就沒辦法及時(shí)喚醒它, 或者永遠(yuǎn)都無法喚醒該線程, 而 interrupt 設(shè)計(jì)之初就是把 wait 等長期阻塞作為一種特殊情況考慮在內(nèi)了, 我們應(yīng)該用 interrupt 思維來停止線程.
以上就是Java 停止線程需要注意的地方的詳細(xì)內(nèi)容,更多關(guān)于Java 停止線程的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!
相關(guān)文章:
1. el-input無法輸入的問題和表單驗(yàn)證失敗問題解決2. 不要在HTML中濫用div3. react腳手架配置代理的實(shí)現(xiàn)4. JavaScript中顏色模型的基礎(chǔ)知識與應(yīng)用詳解5. XML 增、刪、改和查示例6. JavaScript快速實(shí)現(xiàn)一個(gè)顏色選擇器7. CSS3實(shí)例分享之多重背景的實(shí)現(xiàn)(Multiple backgrounds)8. XML入門的常見問題(三)9. 前端html+css實(shí)現(xiàn)動態(tài)生日快樂代碼10. React實(shí)現(xiàn)一個(gè)倒計(jì)時(shí)hook組件實(shí)戰(zhàn)示例
