java中斷線程的正確姿勢完整示例
Java停止線程的邏輯(協(xié)同、通知)
在Java程序中,我們想要停止一個線程可以通過interrupt方法進行停止。但是當(dāng)我們調(diào)用interrupt方法之后,它可能并不會立刻就會停止線程,而是通知線程需要停止。線程接收到通知之后會根據(jù)自身的情況判斷是否需要停止,它可能會立即停止,也有可能會執(zhí)行一段時間后停止,也可能根本就不停止。
那么Java為什么要選擇這種非強制性的線程中斷呢?其實更多是為了數(shù)據(jù)安全,保證程序的健壯性。因為我們不知道程序正在做什么事情。如果貿(mào)然停止,可能會造成數(shù)據(jù)的錯亂、不完整。
一個簡單的例子:
public class _24_ThreadTest implements Runnable {
@Override
public void run() {
int count = 0;
while (!Thread.currentThread().isInterrupted() && count <= 2000) {
System.out.println("count: " + count++);
}
}
public static void main(String[] args) throws Exception {
_24_ThreadTest threadTest = new _24_ThreadTest();
Thread thread = new Thread(threadTest);
thread.start();
Thread.sleep(10);
// 中斷線程
thread.interrupt();
}
}
這個例子是一個簡單的通過interrupt中斷線程的案例,run方法中通過判斷當(dāng)前線程是否中斷,并且count是否大于2000來進行循環(huán)。如果線程中斷則退出循環(huán),線程執(zhí)行結(jié)束。這種就屬于線程正常停止的情況。
Sleep是否會收到線程中斷信號
public class _24_ThreadTest implements Runnable {
@Override
public void run() {
int count = 0;
while (!Thread.currentThread().isInterrupted() && count <= 2000) {
try {
System.out.println("count: " + count++);
// 子線程睡眠
Thread.sleep(1000 * 2);
System.out.println("方法體:" + Thread.currentThread().isInterrupted());
} catch (InterruptedException e) {
System.out.println("異常:" + Thread.currentThread().isInterrupted());
// 線程中斷標志位被重置為false
e.printStackTrace();
}
}
}
public static void main(String[] args) throws Exception {
_24_ThreadTest threadTest = new _24_ThreadTest();
Thread thread = new Thread(threadTest);
thread.start();
Thread.sleep(10);
// 中斷線程
thread.interrupt();
}
}
如果在子線程中睡眠中,主線程通過interrupt方法進行中斷,那么子線程還能不能收到中斷信號。其實在這種情況下線程也是可以接收到信號通知的,這個時候會拋出InterruptedException,并且將線程中斷標志位設(shè)置為false。
在拋出異常后,線程標志位被設(shè)置為false,那么在下次循環(huán)判斷count沒有為false的情況下,還是可以進入循環(huán)體的。這個時候線程就無法停止。
執(zhí)行結(jié)果:

案例場景:
在進行一些后臺任務(wù)通過線程跑的時候,如果在循環(huán)中遇到線程中斷異常,我們需要終止當(dāng)前任務(wù),并且告訴客戶端當(dāng)前任務(wù)執(zhí)行失敗的是哪條記錄,這種情況下就可以通過異常中再次中斷的方式來停止線程,并且可以返回給客戶端當(dāng)前出現(xiàn)異常的記錄是哪條。而不會是接著執(zhí)行下去。
解決方法
public class _24_ThreadTest implements Runnable {
@Override
public void run() {
int count = 0;
while (!Thread.currentThread().isInterrupted() && count <= 2000) {
try {
System.out.println("count: " + count++);
// 子線程睡眠
Thread.sleep(1000 * 2);
Thread.currentThread().interrupt();
System.out.println("方法體:" + Thread.currentThread().isInterrupted());
} catch (InterruptedException e) {
// 再次中斷
Thread.currentThread().interrupt();
System.out.println("異常:" + Thread.currentThread().isInterrupted());
e.printStackTrace();
}
}
}
public static void main(String[] args) throws Exception {
_24_ThreadTest threadTest = new _24_ThreadTest();
Thread thread = new Thread(threadTest);
thread.start();
Thread.sleep(10);
// 中斷線程
thread.interrupt();
}
}
既然我們已經(jīng)知道,在出現(xiàn)線程中斷異常之后線程中斷標志位會被重置為false,那么我們可以在異常中手動的再次中斷當(dāng)前線程,那么就可以完全停止線程任務(wù)。
總結(jié)
上面我們簡單介紹了如何正確的停止線程,如果在以后的面試中被問到這類問題,那么你是不是可以流暢的回答面試官了。
在run方法中遇到異常,我們是不能直接生吞的,一定要做處理,你可以是簡單的日志記錄,也可以中斷線程。但就是不能不做任何處理。
其實還有其他的一些方法來停止線程,比如stop(),這類方法已被舍棄,這種強制停止可能會引起線程的數(shù)據(jù)安全問題,所以已經(jīng)不再推薦使用了。
以上就是java中斷線程的正確姿勢完整示例的詳細內(nèi)容,更多關(guān)于java中斷線程的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Java 中Comparable和Comparator區(qū)別比較
本文,先介紹Comparable 和Comparator兩個接口,以及它們的差異;接著,通過示例,對它們的使用方法進行說明2013-09-09
使用kafka如何選擇分區(qū)數(shù)及kafka性能測試
這篇文章主要介紹了使用kafka如何選擇分區(qū)數(shù)及kafka性能測試,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-08-08
spring?boot配置dubbo方式(properties)
這篇文章主要介紹了spring?boot配置dubbo方式(properties),具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-01-01
Java輸出通過InetAddress獲得的IP地址數(shù)組詳細解析
由于byte被認為是unsigned byte,所以最高位的1將會被解釋為符號位,另外Java中存儲是按照補碼存儲,所以1000 0111會被認為是補碼形式,轉(zhuǎn)換成原碼便是1111 0001,轉(zhuǎn)換成十進制數(shù)便是-1212013-09-09
SpringBoot啟動嵌入式Tomcat的實現(xiàn)步驟
本文主要介紹了淺談SpringBoot如何啟動嵌入式Tomcat,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-08-08
Spring中使用騰訊云發(fā)送短信驗證碼的實現(xiàn)示例
本文主要介紹了Spring?中?使用騰訊云發(fā)送短信驗證碼,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-03-03
springboot中非容器類如何獲取配置文件數(shù)據(jù)
這篇文章主要介紹了springboot中非容器類如何獲取配置文件數(shù)據(jù)問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-01-01

