新聞中心
最近看了點(diǎn) psi-Probe的源代碼,在線程列表頁面,可以對頁面中各個進(jìn)行線程管理,其中有這樣一個操作,見最左側(cè)藍(lán)色方框:

創(chuàng)新互聯(lián)建站是一家專注網(wǎng)站建設(shè)、網(wǎng)絡(luò)營銷策劃、重慶小程序開發(fā)、電子商務(wù)建設(shè)、網(wǎng)絡(luò)推廣、移動互聯(lián)開發(fā)、研究、服務(wù)為一體的技術(shù)型公司。公司成立10年以來,已經(jīng)為成百上千家成都酒店設(shè)計各業(yè)的企業(yè)公司提供互聯(lián)網(wǎng)服務(wù)?,F(xiàn)在,服務(wù)的成百上千家客戶與我們一路同行,見證我們的成長;未來,我們一起分享成功的喜悅。
點(diǎn)擊每個線程對應(yīng)的箭頭按鈕,會彈出下方的提示:
實(shí)際這上按鈕的操作,是要 「Kill」這個指定的線程。
順著鏈接,我們能看到,具體的實(shí)現(xiàn)是這個樣子:
- String threadName = ServletRequestUtils.getStringParameter(request, "thread", null);
- Thread thread = null;
- if (threadName != null) {
- thread = Utils.getThreadByName(threadName);
- }
- if (thread != null) {
- thread.stop();
- }
正如前面的彈窗提示,這里果然調(diào)用的是個危險操作:
- Thread.stop()
這里的 「stop」方法,和「resume」方法、「suspend」方法并稱 Thread 三少,因?yàn)榫€程安全問題,都已經(jīng)被 @Deprecated 了。
官方文檔說的好:
| Stopping a thread causes it to unlock all the monitors that it has locked |
當(dāng)我們停止一個線程時,它會悄悄的把所持有的 monitor 鎖釋放了,此時,其他依賴鎖的線程可能就會搶到鎖執(zhí)行。關(guān)鍵此時,當(dāng)前 stop 的線程實(shí)際并沒有處理完所有先決條件,可能這個時候就產(chǎn)生了詭異的問題,加班的日子可能就悄悄來了。
那你說 「Stop 不讓用了,總得讓我們有辦法處理線程吧,哪怕通知他,打斷他一下,讓他停止」。
目前有以下幾種方式來實(shí)現(xiàn)。
異常
這點(diǎn) Thread 也想到了,提供了一個「異?!箒磉_(dá)到這個打斷的目的。這個異常在其他線程要打斷某個特定線程時執(zhí)行,如果是符合條件,會拋出來。此時這個特定線程自行根據(jù)這次打斷來判斷后續(xù)是不是要再執(zhí)行線程內(nèi)的邏輯,還是直接跳出處理。
這個異常就是 InterruptedException。一般使用方式類似這樣
- try {
- Thread.sleep(backgroundProcessorDelay * 1000L);
- } catch (InterruptedException e) {
- // 具體在中斷通知后的操作
- }
- xxxThread.interrupt();
目前有以下方法能夠進(jìn)行這種操作
- Thread.sleep
- Thread.join
- Object.wait
以wait方法為例,我們來看文檔里的描述
- * @throws InterruptedException if any thread interrupted the
- * current thread before or while the current thread
- * was waiting for a notification. The interrupted
- * status of the current thread is cleared when
- * this exception is thrown.
這里有一點(diǎn)信息: 「interrupted status」,這個是個狀態(tài)標(biāo)識,在Thread類中,可以通過 isInterrupted來判斷當(dāng)前線程是否被中斷。這個標(biāo)識也可以用來作為一個退出線程執(zhí)行的標(biāo)識來直接使用。 但例外是阻塞方法在收到中斷方法調(diào)用后,這個標(biāo)識會被清除重置,所以需要注意下。
我們在執(zhí)行阻塞方法線程的interrupt方法時,此時并不能拿到這個標(biāo)識。
另外,拿到異常時,需要關(guān)注,如果是類似于后臺循環(huán)執(zhí)行的調(diào)度線程,在收到中斷異常時需要處理異常再 break 才能跳出,否則只是相當(dāng)于一個空操作。
目前一些程序里用這種的倒不多,用下面這種的多一些。
退出標(biāo)識
對于一些長駐線程,會在某些時候需要退出執(zhí)行,這種情況下,常采用的操作類似這樣, 以Tomcat 的NioConnector 里的Acceptor為例:
- protected class Acceptor extends AbstractEndpoint.Acceptor {
- @Override
- public void run() {
- int errorDelay = 0;
- // Loop until we receive a shutdown command
- while (running) { // 標(biāo)識1
- // Loop if endpoint is paused
- while (paused && running) { // 標(biāo)識2
- state = AcceptorState.PAUSED;
- try {
- Thread.sleep(50);
- } catch (InterruptedException e) {
- // Ignore
- }
- }
- if (!running) {
- break;
- }
- ...
- }
用這種退出標(biāo)識時,記得一定要聲明為 volatile ,類似這樣:
- /**
- * Running state of the endpoint.
- */
- protected volatile boolean running = false;
- /**
- * Will be set to true whenever the endpoint is paused.
- */
- protected volatile boolean paused = false;
否則因?yàn)槎嗑€程的可見性問題, 這個線程可能一直都不會退出。
目前在 Tomcat 使用中,無法在運(yùn)行時直接操作 Connector ,所以一般情況這個 pause 標(biāo)識可能沒法設(shè)置。但有幾種觸發(fā)的方式,一種是通過 JConsole 等工具連接到 MBeanServer 上,直接通過其MBean方法操作pause,來改變值,另一種是使用類似 psi-Probe(一款功能強(qiáng)大的Tomcat 管理監(jiān)控工具)這種管理控制臺,之前我已經(jīng)把可以操作 Connector 狀態(tài)的代碼提交給 github上(怎樣參與到全世界優(yōu)秀的開源項(xiàng)目中?),commiter 已經(jīng)合入??梢允褂眠M(jìn)行狀態(tài)改變觀察。
總體來說,如果處理sleep/wait等操作,擔(dān)心時間太長,可以通過 interrupt 來進(jìn)行,對于駐留線程,可以通過退出標(biāo)識來處理。
【本文為專欄作者“侯樹成”的原創(chuàng)稿件,轉(zhuǎn)載請通過作者微信公眾號『Tomcat那些事兒』獲取授權(quán)】
本文標(biāo)題:如何優(yōu)雅的「打斷」你的線程?
本文地址:http://fisionsoft.com.cn/article/ccdhiis.html


咨詢
建站咨詢
