java高并發(fā)的線程中斷的幾種方式詳解
通過一個變量控制線程中斷
代碼:
package com.itsoku.chat05;
import java.util.concurrent.TimeUnit;
/**
* 微信公眾號:路人甲Java,專注于java技術(shù)分享(帶你玩轉(zhuǎn) 爬蟲、分布式事務(wù)、異步消息服務(wù)、任務(wù)調(diào)度、分庫分表、大數(shù)據(jù)等),喜歡請關(guān)注!
*/
public class Demo1 {
public volatile static boolean exit = false;
public static class T extends Thread {
@Override
public void run() {
while (true) {
//循環(huán)處理業(yè)務(wù)
if (exit) {
break;
}
}
}
}
public static void setExit() {
exit = true;
}
public static void main(String[] args) throws InterruptedException {
T t = new T();
t.start();
TimeUnit.SECONDS.sleep(3);
setExit();
}
}
代碼中啟動了一個線程,線程的run方法中有個死循環(huán),內(nèi)部通過exit變量的值來控制是否退出。 TimeUnit.SECONDS.sleep(3);讓主線程休眠3秒,此處為什么使用TimeUnit?TimeUnit使用更方便一些,能夠很清晰的控制休眠時間,底層還是轉(zhuǎn)換為Thread.sleep實現(xiàn)的。程序有個重點:volatile關(guān)鍵字,exit變量必須通過這個修飾,如果把這個去掉,程序無法正常退出。volatile控制了變量在多線程中的可見性,關(guān)于volatile前面的文章中有介紹,此處就不再說了。
通過線程自帶的中斷標(biāo)志控制
示例代碼:
package com.itsoku.chat05;
import java.util.concurrent.TimeUnit;
/**
* 微信公眾號:路人甲Java,專注于java技術(shù)分享(帶你玩轉(zhuǎn) 爬蟲、分布式事務(wù)、異步消息服務(wù)、任務(wù)調(diào)度、分庫分表、大數(shù)據(jù)等),喜歡請關(guān)注!
*/
public class Demo2 {
public static class T extends Thread {
@Override
public void run() {
while (true) {
//循環(huán)處理業(yè)務(wù)
if (this.isInterrupted()) {
break;
}
}
}
}
public static void main(String[] args) throws InterruptedException {
T t = new T();
t.start();
TimeUnit.SECONDS.sleep(3);
t.interrupt();
}
}
運行上面的程序,程序可以正常結(jié)束。線程內(nèi)部有個中斷標(biāo)志,當(dāng)調(diào)用線程的interrupt()實例方法之后,線程的中斷標(biāo)志會被置為true,可以通過線程的實例方法isInterrupted()獲取線程的中斷標(biāo)志。
線程阻塞狀態(tài)中如何中斷?
示例代碼:
package com.itsoku.chat05;
import java.util.concurrent.TimeUnit;
/**
* 微信公眾號:路人甲Java,專注于java技術(shù)分享(帶你玩轉(zhuǎn) 爬蟲、分布式事務(wù)、異步消息服務(wù)、任務(wù)調(diào)度、分庫分表、大數(shù)據(jù)等),喜歡請關(guān)注!
*/
public class Demo3 {
public static class T extends Thread {
@Override
public void run() {
while (true) {
//循環(huán)處理業(yè)務(wù)
//下面模擬阻塞代碼
try {
TimeUnit.SECONDS.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
public static void main(String[] args) throws InterruptedException {
T t = new T();
t.start();
}
}
運行上面代碼,發(fā)現(xiàn)程序無法結(jié)束。
在此先補充幾點知識:
1.調(diào)用線程的interrupt()實例方法,線程的中斷標(biāo)志會被置為true
2.當(dāng)線程處于阻塞狀態(tài)時,調(diào)用線程的interrupt()實例方法,線程內(nèi)部會觸發(fā)InterruptedException異常,并且會清除線程內(nèi)部的中斷標(biāo)志(即將中斷標(biāo)志置為false)
那么上面代碼可以調(diào)用線程的interrupt()方法來引發(fā)InterruptedException異常,來中斷sleep方法導(dǎo)致的阻塞,調(diào)整一下代碼,如下:
package com.itsoku.chat05;
import java.util.concurrent.TimeUnit;
/**
* 微信公眾號:路人甲Java,專注于java技術(shù)分享(帶你玩轉(zhuǎn) 爬蟲、分布式事務(wù)、異步消息服務(wù)、任務(wù)調(diào)度、分庫分表、大數(shù)據(jù)等),喜歡請關(guān)注!
*/
public class Demo3 {
public static class T extends Thread {
@Override
public void run() {
while (true) {
//循環(huán)處理業(yè)務(wù)
//下面模擬阻塞代碼
try {
TimeUnit.SECONDS.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
this.interrupt();
}
if (this.isInterrupted()) {
break;
}
}
}
}
public static void main(String[] args) throws InterruptedException {
T t = new T();
t.start();
TimeUnit.SECONDS.sleep(3);
t.interrupt();
}
}
運行結(jié)果:
java.lang.InterruptedException: sleep interrupted
at java.lang.Thread.sleep(Native Method)
at java.lang.Thread.sleep(Thread.java:340)
at java.util.concurrent.TimeUnit.sleep(TimeUnit.java:386)
at com.itsoku.chat05.Demo3$T.run(Demo3.java:17)
程序可以正常結(jié)束了,分析一下上面代碼,注意幾點:
1.main方法中調(diào)用了t.interrupt()方法,此時線程t內(nèi)部的中斷標(biāo)志會置為true
2.然后會觸發(fā)run()方法內(nèi)部的InterruptedException異常,所以運行結(jié)果中有異常輸出,上面說了,當(dāng)觸發(fā)InterruptedException異常時候,線程內(nèi)部的中斷標(biāo)志又會被清除(變?yōu)閒alse),所以在catch中又調(diào)用了this.interrupt();一次,將中斷標(biāo)志置為false
3.run()方法中通過this.isInterrupted()來獲取線程的中斷標(biāo)志,退出循環(huán)(break)
總結(jié)
當(dāng)一個線程處于被阻塞狀態(tài)或者試圖執(zhí)行一個阻塞操作時,可以使用 Thread.interrupt()方式中斷該線程,注意此時將會拋出一個InterruptedException的異常,同時中斷狀態(tài)將會被復(fù)位(由中斷狀態(tài)改為非中斷狀態(tài))
內(nèi)部有循環(huán)體,可以通過一個變量來作為一個信號控制線程是否中斷,注意變量需要volatile修飾
文中的幾種方式可以結(jié)合起來靈活使用控制線程的中斷
本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
MybatisPlus lambdaQueryWrapper中常用方法的使用
本文主要介紹了MybatisPlus lambdaQueryWrapper中常用方法的使用,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-07-07
使用SpringEvent解決WebUploader大文件上傳解耦問題
Spring Event是Spring框架內(nèi)建的一種發(fā)布/訂閱模式的實現(xiàn),它允許應(yīng)用內(nèi)部不同組件之間通過事件進行通信,本文以WebUploader大文件上傳組件為例,在大文件處理的場景中使用SpringEvent的事件發(fā)布機制,靈活的擴展對文件的處理需求,需要的朋友可以參考下2024-07-07
Spring?Boot?使用斷言讓你的代碼在上線前就通過“體檢”(最新整理)
斷言是一種編程技巧,用于在代碼中插入檢查點,驗證程序的狀態(tài)是否符合預(yù)期,如果斷言失敗,程序會拋出一個錯誤,幫助你快速發(fā)現(xiàn)和修復(fù)bug,本文給大家介紹Spring?Boot?斷言:讓你的代碼在上線前就通過“體檢”,感興趣的朋友一起看看吧2025-03-03
java中的Timer和Timertask的關(guān)系解讀
本文詳細介紹了Java中的Timer和TimerTask類,包括它們之間的關(guān)系、API的使用方法、注意事項以及操作案例,Timer是一個調(diào)度器,而TimerTask是具體的任務(wù)類,Timer僅對應(yīng)一個線程,不保證任務(wù)執(zhí)行的精確性,但線程安全,一個Timer可以調(diào)度多個TimerTask2024-12-12
CountDownLatch源碼解析之a(chǎn)wait()
這篇文章主要為大家詳細解析了CountDownLatch源碼之a(chǎn)wait方法,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-04-04

