Java 正確終止線程的方法
Thread類中有一個(gè)已經(jīng)廢棄的 stop() 方法,它可以終止線程,但由于它不管三七二十一,直接終止線程,所以被廢棄了。比如,當(dāng)線程被停止后還需要進(jìn)行一些善后操作(如,關(guān)閉外部資源),使用這個(gè)方法就無能為力了。可以通過線程中斷來實(shí)現(xiàn)線程終止。
首先來看一下Java線程中斷的一些內(nèi)容:
- Java平臺(tái)為每個(gè)線程維護(hù)了一個(gè)布爾型的中斷標(biāo)記,可以通過下列方法獲取該標(biāo)記的值:
interrupt() 中斷某個(gè)線程
isInterrupted() 返回該線程的中斷標(biāo)記
interrupted() 返回并重置該線程的中斷標(biāo)記(置為false)
- 中斷僅是發(fā)起線程對(duì)目標(biāo)線程的一種請(qǐng)求,也就是說,目標(biāo)線程對(duì)這種請(qǐng)求可以相應(yīng),也可以忽略。
- Java標(biāo)準(zhǔn)庫中與線程阻塞相關(guān)的方法對(duì)中斷的相應(yīng)方式都是拋出 InterruptedException 異常,并且按照慣例,拋出異常前都會(huì)重置中斷標(biāo)記為false,因此這些方法會(huì)清空線程的中斷標(biāo)記。
- Java標(biāo)準(zhǔn)庫中與線程阻塞相關(guān)的方法在進(jìn)行阻塞前會(huì)判斷中斷標(biāo)記是否為true,為true則拋出異常;如果在阻塞后調(diào)用中斷方法的話,那么JVM會(huì)設(shè)置該線程的中斷標(biāo)記,然后將該線程喚醒,因此中斷具有喚醒線程的作用。
由上面幾點(diǎn)和第二句加粗的話可知,可以使用線程中斷來實(shí)現(xiàn)線程終止,只要目標(biāo)線程判斷一下中斷標(biāo)記即可,即使被中斷的線程正處于阻塞狀態(tài),也能把他喚醒起來終止;由第一句加粗的話可知,直接使用線程中斷實(shí)現(xiàn)線程終止是存在風(fēng)險(xiǎn)的,因?yàn)榭赡苷{(diào)用了一些Java標(biāo)準(zhǔn)庫的阻塞方法,而導(dǎo)致了中斷標(biāo)記被清空,也就無法獲得中斷標(biāo)記了(總是false),因此需要自己創(chuàng)建一個(gè)中斷標(biāo)記配合使用。
如,下面是一個(gè)可中斷的任務(wù)執(zhí)行器,他會(huì)在每次執(zhí)行任務(wù)前,判斷一下自定i的終止標(biāo)記和剩余的任務(wù)數(shù)(善后);提供的shutdown方法除了將工作線程中斷外(主要作用是喚醒可能處于阻塞狀態(tài)的任務(wù)),還會(huì)將終止交集 terminated 置為 true。
執(zhí)行 main 方法,可以發(fā)現(xiàn),首先會(huì)打印出“客戶端調(diào)用了 shutdown 方法”,然后過了四秒,main線程才會(huì)終止,可知shutdown方法正確地將目標(biāo)線程終止了。關(guān)于“按照慣例,Java標(biāo)準(zhǔn)庫中拋出InterruptedException異常的和線程相關(guān)的阻塞方法會(huì)清空中斷標(biāo)記”,可以將條件中的 !interminated 替換成 !Thread.currentThread().isInterrupted(),然后再執(zhí)行main方法測(cè)試,可以發(fā)現(xiàn)main線程始終無法終止,因?yàn)?sleep() 方法清空了中斷標(biāo)記,所以 !Thread.currentThread().isInterrupted() 始終為true,導(dǎo)致工作線程始終無法終止。
public class TerminableTaskRunner {
// 存儲(chǔ)要執(zhí)行的任務(wù)
private final BlockingQueue<Runnable> tasks;
// 線程終止標(biāo)志
private volatile boolean terminated;
// 剩余的任務(wù)數(shù)
private final AtomicInteger count;
// 實(shí)際執(zhí)行任務(wù)的線程
private volatile Thread workThread;
public TerminableTaskRunner(int capacity) {
this.tasks = new LinkedBlockingDeque<>(capacity);
this.count = new AtomicInteger(0);
this.workThread = new WorkThread();
workThread.start();
}
public void submit(Runnable task) {
this.tasks.add(task);
this.count.incrementAndGet();
}
public void shutdown() {
terminated = true; // 線程終止標(biāo)志,由于中斷標(biāo)志可能會(huì)被覆蓋,所以需要自己創(chuàng)建一個(gè)標(biāo)志
if (workThread != null)
workThread.interrupt(); // 喚醒線程
}
private class WorkThread extends Thread {
@Override
public void run() {
Runnable task;
try {
while (!terminated || tasks.size() >= 1) {
task = tasks.take();
try {
task.run(); // 可能會(huì)清空當(dāng)前線程的中斷標(biāo)記,如task.run()在內(nèi)部調(diào)用的阻塞方法拋出了InterruptedException
} catch (Throwable e) {
e.printStackTrace();
}
count.decrementAndGet();
}
} catch (InterruptedException e) {
// 一旦調(diào)用shutdown且tasks.take()阻塞住,就拋出該異常,沒有任務(wù)要執(zhí)行,直接終止
workThread = null;
}
}
}
public static void main(String[] args) {
TerminableTaskRunner taskRunner = new TerminableTaskRunner(4);
for (int i = 0; i < 4; i++) {
taskRunner.submit(()->{
try {
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
System.out.println("客戶端調(diào)用了 shutdown 方法");
}
});
}
taskRunner.shutdown();
}
}
以上就是Java 正確終止線程的方法的詳細(xì)內(nèi)容,更多關(guān)于Java 終止線程的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
基于HTML5+js+Java實(shí)現(xiàn)單文件文件上傳到服務(wù)器功能
應(yīng)公司要求,在HTML5頁面上實(shí)現(xiàn)上傳文件到服務(wù)器功能,對(duì)于我這樣的菜鳥,真是把我難住了,最后還是請(qǐng)教大神搞定的,下面小編把例子分享到腳本之家平臺(tái),供大家參考2017-08-08
SpringBoot文件上傳(本地存儲(chǔ))回顯前端操作方法
這篇文章主要介紹了SpringBoot文件上傳(本地存儲(chǔ))回顯前端操作方法的相關(guān)資料,文中講解了文件上傳的基本原理,包括前端調(diào)用后端接口上傳文件,后端返回文件路徑給前端,前端通過路徑訪問圖片,需要的朋友可以參考下2024-11-11
Java跳臺(tái)階實(shí)現(xiàn)思路和代碼
今天小編就為大家分享一篇關(guān)于Java跳臺(tái)階實(shí)現(xiàn)思路和代碼,小編覺得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來看看吧2019-01-01
Spring事務(wù)aftercommit原理及實(shí)踐
這篇文章主要為大家介紹了Spring事務(wù)aftercommit原理及實(shí)踐,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-09-09
Java后端學(xué)習(xí)精華之TCP通信傳輸協(xié)議詳解
TCP/IP是一種面向連接的、可靠的、基于字節(jié)流的傳輸層通信協(xié)議,它會(huì)保證數(shù)據(jù)不丟包、不亂序。TCP全名是Transmission Control Protocol,它是位于網(wǎng)絡(luò)OSI模型中的第四層2021-09-09
Java中關(guān)于Collections集合工具類的詳細(xì)介紹
Java提供了一個(gè)操作Set、List和Map等集合的工具類:Collections,該工具提供了大量方法對(duì)集合元素進(jìn)行排序、查詢和修改等操作,還提供了將集合對(duì)象設(shè)置為不可變、對(duì)集合對(duì)象實(shí)現(xiàn)同步控制等方法2021-09-09
jvm中指定時(shí)區(qū)信息user.timezone問題及解決方式
同一份程序使用時(shí)間LocalDateTime類型,在國內(nèi)和國外部署后,返回的時(shí)間信息前端使用出問題,這篇文章主要介紹了jvm中指定時(shí)區(qū)信息user.timezone問題及解決方法,需要的朋友可以參考下2023-02-02

