淺談Java線程Thread之interrupt中斷解析
這一篇我們說說Java線程Thread的interrupt中斷機(jī)制。
中斷線程
線程的thread.interrupt()方法是中斷線程,將會(huì)設(shè)置該線程的中斷狀態(tài)位,即設(shè)置為true,中斷的結(jié)果線程是死亡、還是等待新的任務(wù)或是繼續(xù)運(yùn)行至下一步,就取決于這個(gè)程序本身。線程會(huì)不時(shí)地檢測(cè)這個(gè)中斷標(biāo)示位,以判斷線程是否應(yīng)該被中斷(中斷標(biāo)示值是否為true)。它并不像stop方法那樣會(huì)中斷一個(gè)正在運(yùn)行的線程。
判斷線程是否被中斷
判斷某個(gè)線程是否已被發(fā)送過中斷請(qǐng)求,請(qǐng)使用Thread.currentThread().isInterrupted()方法(因?yàn)樗鼘⒕€程中斷標(biāo)示位設(shè)置為true后,不會(huì)立刻清除中斷標(biāo)示位,即不會(huì)將中斷標(biāo)設(shè)置為false),而不要使用thread.interrupted()(該方法調(diào)用后會(huì)將中斷標(biāo)示位清除,即重新設(shè)置為false)方法來判斷,下面是線程在循環(huán)中時(shí)的中斷方式:
while(!Thread.currentThread().isInterrupted() && more work to do){ do more work }
interrupt之中斷狀態(tài)標(biāo)記
interrupt中斷機(jī)制中有如下方法:
- Thread.interrupt(),設(shè)置當(dāng)前中斷標(biāo)記為true(類似屬性的set方法)
- Thread.isInterrupted(),檢測(cè)當(dāng)前的中斷標(biāo)記(類似屬性的get方法)
- Thread.interrupted(),檢測(cè)當(dāng)前的中斷標(biāo)記,然后重置中斷標(biāo)記為false(類似屬性的get方法+set方法)
因此interrupt中斷機(jī)制并不是真正的將當(dāng)前線程中斷,而是一個(gè)中斷標(biāo)記的變化。我們先用例子來測(cè)試一下。
public class InterruptTest { //這里用來打印消耗的時(shí)間 private static long time = 0; private static void resetTime(){ time = System.currentTimeMillis(); } private static void printContent(String content){ System.out.println(content + " 時(shí)間:" + (System.currentTimeMillis() - time)); } public static void main(String[] args) { test1(); } private static void test1(){ Thread1 thread1 = new Thread1(); thread1.start(); //延時(shí)3秒后interrupt中斷 try { Thread.sleep(3000); } catch (InterruptedException e) { e.printStackTrace(); } thread1.interrupt(); printContent("執(zhí)行中斷"); } private static class Thread1 extends Thread{ @Override public void run() { resetTime(); int num = 0; while (true){ if(isInterrupted()){ printContent("當(dāng)前線程 isInterrupted"); break; } num++; if(num % 100 == 0){ printContent("num : " + num); } } } } }
以上代碼是開啟一個(gè)Thread1線程,在Thread1線程的while循環(huán)中不斷對(duì)num加1,每到100的倍數(shù)打印一次(防止打印太快)。然后主線程在sleep了3000毫秒后,調(diào)用Thread1線程的interrupt方法。那么我們看看輸出結(jié)果:
intterupt中斷
可以看到,在耗時(shí)3000毫秒左右,也就是主線程sleep之后執(zhí)行thread1.interrupt();后,Thread1線程停止了,而Thread1線程的停止是因?yàn)閣hile循環(huán)中的isInterrupted方法返回了true,所以break退出了while循環(huán),也就是說interrupt和isInterrupted在這里起到的作用就相當(dāng)于setXX和getXX的作用,維護(hù)著一個(gè)boolean變量。
interrupt之中斷異常處理
當(dāng)然interrupt機(jī)制并不僅僅是一個(gè)中斷狀態(tài)位的變化和檢測(cè),它還可以進(jìn)行中斷異常的處理。我們知道Thread.sleep()方法需要捕獲中斷異常,那接下來我們往其中添加一個(gè)sleep延時(shí)試試
while (true){ if(isInterrupted()){ printContent("當(dāng)前線程 isInterrupted"); break; } num++; //sleep一下 try { Thread.sleep(1); } catch (InterruptedException e) { e.printStackTrace(); } if(num % 100 == 0){ printContent("num : " + num); } }
我們?cè)倏纯摧敵鼋Y(jié)果:
intterupt中斷
這里我們會(huì)發(fā)現(xiàn),sleep睡眠之后,輸出的num值明顯小了好多(沒睡眠時(shí)num都達(dá)到10億的大小了,看來CPU執(zhí)行簡(jiǎn)單運(yùn)算還是非常快的),哈哈,不過這不是重點(diǎn),重點(diǎn)是是看到輸出了一個(gè)異常,還有就是輸出異常后,isInterrupted輸出返回false,Thread1線程又繼續(xù)執(zhí)行下去了,并沒有退出while循環(huán)。那么這是為什么呢?我們只是加了一個(gè)sleep睡眠而已。
如果Thread1線程中有執(zhí)行需要捕獲InterruptedException異常的操作,比如Thread的sleep,join方法,Object的wait,Condition的await等,它是強(qiáng)制需要捕獲InterruptedException異常的,那么當(dāng)thread1.interrupt方法調(diào)用之后,它會(huì)給thread1線程拋出一個(gè)InterruptedException異常,那么在while循環(huán)中,就能捕獲到這個(gè)異常然后這個(gè)異常拋出之后,又會(huì)馬上將線程中斷標(biāo)識(shí)重置為false,因此在下次的while循環(huán)中判斷isInterrupted時(shí),它是false,也就不會(huì)break,然后while循環(huán)會(huì)一直執(zhí)行下去。
因此interrupt()方法會(huì)根據(jù)thread線程中的run方法里是否有必須捕獲InterruptedException異常的代碼,而做出不同操作:
- 如果沒有必須捕獲InterruptedException異常的代碼(比如Thread.sleep()),則isInterrupted()會(huì)返回true,此時(shí)可以在isInterrupted的判斷中處理中斷變化。
- 如果有必須捕獲InterruptedException異常的代碼(比如Thread.sleep()),則會(huì)拋出InterruptedException異常,并進(jìn)行捕獲,同時(shí)重置isInterrupted為false,此時(shí)得在異常捕獲中處理中斷變化。
interrupt的應(yīng)用場(chǎng)景
通常interrupt適用于在線程執(zhí)行中的循環(huán)標(biāo)記判斷,例如
while(!isInterrupted()){ ... }
但是如果在本次循環(huán)中出現(xiàn)阻塞了,那么線程就無法判斷下次的isInterrupted標(biāo)記,那么即便調(diào)用了interrupt()方法也無法退出循環(huán),也就無法退出線程。例如
while(!isInterrupted()){ ... while(true){ //線程卡在這里了,則無法響應(yīng)interrupte機(jī)制了 } }
這樣的話,interrupt就沒轍了,線程會(huì)一直執(zhí)行下去,不會(huì)被中斷停止。
測(cè)試?yán)硬榭?我的GitHub--JavaTest
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Mybatis中關(guān)于自定義mapper.xml時(shí),參數(shù)傳遞的方式及寫法
這篇文章主要介紹了Mybatis中關(guān)于自定義mapper.xml時(shí),參數(shù)傳遞的方式及寫法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-12-12Java中實(shí)現(xiàn) SHA-256加密的兩種方式
這篇文章主要介紹了Java中實(shí)現(xiàn) SHA-256加密的兩種方式,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2024-01-01Java使用JMeter進(jìn)行高并發(fā)測(cè)試
軟件的壓力測(cè)試是一種保證軟件質(zhì)量的行為,本文主要介紹了Java使用JMeter進(jìn)行高并發(fā)測(cè)試,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-11-11詳解SpringMVC中的四種跳轉(zhuǎn)方式、視圖解析器問題
這篇文章主要介紹了SpringMVC的四種跳轉(zhuǎn)方式、視圖解析器,springmvc核心配置文件和視圖解析器的使用,添加視圖解析器,通過案例講解四種跳轉(zhuǎn)方式,需要的朋友可以參考下2022-10-10JAVA設(shè)置手動(dòng)提交事務(wù),回滾事務(wù),提交事務(wù)的操作
這篇文章主要介紹了JAVA設(shè)置手動(dòng)提交事務(wù),回滾事務(wù),提交事務(wù)的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-04-04