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