Java中實(shí)現(xiàn)線程的超時(shí)中斷方法實(shí)例
背景
之前在實(shí)現(xiàn)熔斷降級(jí)組件時(shí),需要實(shí)現(xiàn)一個(gè)接口的超時(shí)中斷,意思是,業(yè)務(wù)在使用熔斷降級(jí)功能時(shí),在平臺(tái)上設(shè)置了一個(gè)超時(shí)時(shí)間,如果在請(qǐng)求進(jìn)入熔斷器開(kāi)始計(jì)時(shí),并且接口在超時(shí)時(shí)間內(nèi)沒(méi)有響應(yīng),則需要提早中斷該請(qǐng)求并返回。
比如正常下游接口的超時(shí)時(shí)間為800ms,但是因?yàn)樽陨順I(yè)務(wù)的特殊需求,最多只能等200ms,如果200ms之內(nèi)沒(méi)有數(shù)據(jù)返回,則返回降級(jí)數(shù)據(jù)。這里處理請(qǐng)求的線程可以看成是tomcat線程池中的一個(gè)線程,如果通過(guò)線程池返回的Future,可以很輕松的實(shí)現(xiàn)超時(shí)返回,但是這種情況下,并不能拿到Futrue,需要換一種思路。
思路
中斷一個(gè)線程的思路有哪些?
除了已經(jīng)廢棄的Thread.stop, Thread.suspend, Thread.resume 方法,剩下的貌似只有一種方案了,就是調(diào)用當(dāng)前線程的 interrupt() ,但是這個(gè)方法的作用并不是中斷線程,而是設(shè)置一個(gè)標(biāo)識(shí),通知該線程可以被中斷了,到底是繼續(xù)執(zhí)行,還是中斷返回,由線程本身自己決定。
具體來(lái)說(shuō),當(dāng)對(duì)一個(gè)線程調(diào)用了 interrupt() 之后,如果該線程處于被阻塞狀態(tài)(比如執(zhí)行了wait、sleep或join等方法),那么會(huì)立即退出阻塞狀態(tài),并拋出一個(gè) InterruptedException 異常,在代碼中catch這個(gè)異常進(jìn)行后續(xù)處理。如果線程一直處于運(yùn)行狀態(tài),那么只會(huì)把該線程的中斷標(biāo)志設(shè)置為 true,僅此而已,所以 interrupt() 并不能真正的中斷線程,不過(guò)在rpc調(diào)用的場(chǎng)景中,請(qǐng)求線程一般都處于阻塞狀態(tài),等待數(shù)據(jù)返回,這時(shí) interrupt() 方法是可以派上用場(chǎng)的。
那么,要實(shí)現(xiàn)指定超時(shí)時(shí)間內(nèi)中斷請(qǐng)求線程,還有最后一個(gè)問(wèn)題需要解決:什么時(shí)候,由誰(shuí)去執(zhí)行 interrupt() 方法?
必然這個(gè)方法只能由其它線程來(lái)執(zhí)行了(自己都阻塞了,執(zhí)行個(gè)鬼),而且是在請(qǐng)求進(jìn)入熔斷器時(shí),并在超時(shí)時(shí)間之后執(zhí)行,有點(diǎn)繞,比如超時(shí)時(shí)間是200ms,那么請(qǐng)求進(jìn)入熔斷器之后,再過(guò)200ms,就執(zhí)行 interrupt() ,但是在200ms之內(nèi)有數(shù)據(jù)返回,那么就不執(zhí)行 interrupt() 了。
實(shí)現(xiàn)
需求已經(jīng)很明確了,相當(dāng)于延遲執(zhí)行一個(gè)task,其內(nèi)部邏輯就是執(zhí)行請(qǐng)求線程的 interrupt() ,當(dāng)然還有其它的邏輯。
Runnable task = new Runnable() {
@Override
public void run() {
try {
thread.interrupt();
// 取消定時(shí)器任務(wù)
f.cancel();
} catch (Exception e) {
logger.error("Failed while ticking TimerListener", e);
}
}
};
Doug Lea大神提供的 ScheduledThreadPoolExecutor 可以很好的滿足這個(gè)需求,通過(guò) scheduleAtFixedRate 方法可以很方便的實(shí)現(xiàn)在延遲指定時(shí)間之后執(zhí)行提交的任務(wù)。
ScheduledFuture<?> f = executor.scheduleAtFixedRate( task, timeout, timeout, TimeUnit.MILLISECONDS);
在請(qǐng)求進(jìn)入熔斷器時(shí),順便提交一個(gè)任務(wù)到線程池中等待執(zhí)行,如果接口在超時(shí)時(shí)間內(nèi)沒(méi)有返回,那么該任務(wù)會(huì)被觸發(fā),并執(zhí)行請(qǐng)求線程的 interrupt 方法,這樣就實(shí)現(xiàn)了請(qǐng)求線程的中斷(因?yàn)檫@時(shí)請(qǐng)求線程正在被阻塞,等待數(shù)據(jù)返回),另外需要清空定時(shí)任務(wù),不然這個(gè)任務(wù)會(huì)一直執(zhí)行。
如果接口正常返回了,也要記得清空定時(shí)任務(wù),并且在請(qǐng)求退出熔斷器的時(shí)候,記得恢復(fù)請(qǐng)求線程的中斷標(biāo)識(shí),如何恢復(fù)?在請(qǐng)求線程中執(zhí)行下面代碼即可。
Thread.interrupted();
// 內(nèi)部邏輯
public static boolean interrupted() {
return currentThread().isInterrupted(true);
}
// 參數(shù)為true,可以清除中斷標(biāo)識(shí)
private native boolean isInterrupted(boolean ClearInterrupted);
執(zhí)行當(dāng)前線程(即請(qǐng)求線程)的isInterrupted方法。
使用這種方式實(shí)現(xiàn)請(qǐng)求的超時(shí)中斷,在QPS很高的情況下,會(huì)有額外的性能損失,因?yàn)槊看握?qǐng)求都要提交一個(gè)任務(wù)到線程池中等待執(zhí)行。
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
Java Tree結(jié)構(gòu)數(shù)據(jù)中查找匹配節(jié)點(diǎn)方式
這篇文章主要介紹了Java Tree結(jié)構(gòu)數(shù)據(jù)中查找匹配節(jié)點(diǎn)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-09-09
基于MyBatis的parameterType傳入?yún)?shù)類(lèi)型
這篇文章主要介紹了基于MyBatis的parameterType傳入?yún)?shù)類(lèi)型,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09
Apache?Commons?BeanUtils:?JavaBean操作方法
這篇文章主要介紹了Apache?Commons?BeanUtils:?JavaBean操作的藝術(shù),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-12-12
Java使用通配符實(shí)現(xiàn)增強(qiáng)泛型詳解
泛型是JAVA重要的特性,使用泛型編程,可以使代碼復(fù)用率提高。本文將利用通配符實(shí)現(xiàn)增強(qiáng)泛型,文中的示例代碼講解詳細(xì),感興趣的可以了解一下2022-08-08
java注解實(shí)現(xiàn)websocket服務(wù)的兩種方式
Java WebSocket是一種基于TCP協(xié)議的雙向全雙工消息傳輸技術(shù),它允許服務(wù)器和客戶端之間實(shí)時(shí)通信,具有低延遲和高效率的特點(diǎn),下面這篇文章主要給大家介紹了關(guān)于java注解實(shí)現(xiàn)websocket服務(wù)的兩種方式,需要的朋友可以參考下2024-08-08
Spring Boot 實(shí)現(xiàn)圖片上傳并回顯功能
本篇文章給大家分享Spring Boot 實(shí)現(xiàn)圖片上傳并回顯功能,文中通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2021-07-07
Java創(chuàng)建student類(lèi)詳細(xì)代碼例子
這篇文章主要給大家介紹了關(guān)于Java創(chuàng)建student類(lèi)的相關(guān)資料,學(xué)生類(lèi)(Student)是一種面向?qū)ο蟮木幊谈拍?其主要用于描述學(xué)生的屬性和行為,文中通過(guò)代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-11-11
Java數(shù)組轉(zhuǎn)換為集合的相關(guān)方法
在Java中我們經(jīng)常需要將數(shù)組從一種類(lèi)型轉(zhuǎn)換為另一種類(lèi)型,下面這篇文章主要給大家介紹了關(guān)于Java數(shù)組轉(zhuǎn)換為集合的相關(guān)方法,文中通過(guò)圖文及代碼介紹的非常詳細(xì),需要的朋友可以參考下2024-01-01
使用MyBatisPlus自動(dòng)生成代碼后tomcat運(yùn)行報(bào)錯(cuò)的問(wèn)題及解決方法
這篇文章主要介紹了使用MyBatisPlus自動(dòng)生成代碼后tomcat運(yùn)行報(bào)錯(cuò)的問(wèn)題及解決方法,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-08-08

