通過一個(gè)變量控制線程中斷
代碼:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
|
package com.itsoku.chat05; import java.util.concurrent.TimeUnit; /** * 微信公眾號(hào):路人甲Java,專注于java技術(shù)分享(帶你玩轉(zhuǎn) 爬蟲、分布式事務(wù)、異步消息服務(wù)、任務(wù)調(diào)度、分庫分表、大數(shù)據(jù)等),喜歡請(qǐng)關(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(); } } |
代碼中啟動(dòng)了一個(gè)線程,線程的run方法中有個(gè)死循環(huán),內(nèi)部通過exit變量的值來控制是否退出。 TimeUnit.SECONDS.sleep(3);讓主線程休眠3秒,此處為什么使用TimeUnit?TimeUnit使用更方便一些,能夠很清晰的控制休眠時(shí)間,底層還是轉(zhuǎn)換為Thread.sleep實(shí)現(xiàn)的。程序有個(gè)重點(diǎn):volatile關(guān)鍵字,exit變量必須通過這個(gè)修飾,如果把這個(gè)去掉,程序無法正常退出。volatile控制了變量在多線程中的可見性,關(guān)于volatile前面的文章中有介紹,此處就不再說了。
通過線程自帶的中斷標(biāo)志控制
示例代碼:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
|
package com.itsoku.chat05; import java.util.concurrent.TimeUnit; /** * 微信公眾號(hào):路人甲Java,專注于java技術(shù)分享(帶你玩轉(zhuǎn) 爬蟲、分布式事務(wù)、異步消息服務(wù)、任務(wù)調(diào)度、分庫分表、大數(shù)據(jù)等),喜歡請(qǐng)關(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(); } } |
運(yùn)行上面的程序,程序可以正常結(jié)束。線程內(nèi)部有個(gè)中斷標(biāo)志,當(dāng)調(diào)用線程的interrupt()實(shí)例方法之后,線程的中斷標(biāo)志會(huì)被置為true,可以通過線程的實(shí)例方法isInterrupted()獲取線程的中斷標(biāo)志。
線程阻塞狀態(tài)中如何中斷?
示例代碼:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
|
package com.itsoku.chat05; import java.util.concurrent.TimeUnit; /** * 微信公眾號(hào):路人甲Java,專注于java技術(shù)分享(帶你玩轉(zhuǎn) 爬蟲、分布式事務(wù)、異步消息服務(wù)、任務(wù)調(diào)度、分庫分表、大數(shù)據(jù)等),喜歡請(qǐng)關(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(); } } |
運(yùn)行上面代碼,發(fā)現(xiàn)程序無法結(jié)束。
在此先補(bǔ)充幾點(diǎn)知識(shí):
1.調(diào)用線程的interrupt()實(shí)例方法,線程的中斷標(biāo)志會(huì)被置為true
2.當(dāng)線程處于阻塞狀態(tài)時(shí),調(diào)用線程的interrupt()實(shí)例方法,線程內(nèi)部會(huì)觸發(fā)InterruptedException異常,并且會(huì)清除線程內(nèi)部的中斷標(biāo)志(即將中斷標(biāo)志置為false)
那么上面代碼可以調(diào)用線程的interrupt()方法來引發(fā)InterruptedException異常,來中斷sleep方法導(dǎo)致的阻塞,調(diào)整一下代碼,如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
|
package com.itsoku.chat05; import java.util.concurrent.TimeUnit; /** * 微信公眾號(hào):路人甲Java,專注于java技術(shù)分享(帶你玩轉(zhuǎn) 爬蟲、分布式事務(wù)、異步消息服務(wù)、任務(wù)調(diào)度、分庫分表、大數(shù)據(jù)等),喜歡請(qǐng)關(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(); } } |
運(yùn)行結(jié)果:
1
2
3
4
5
|
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é)束了,分析一下上面代碼,注意幾點(diǎn):
1.main方法中調(diào)用了t.interrupt()方法,此時(shí)線程t內(nèi)部的中斷標(biāo)志會(huì)置為true
2.然后會(huì)觸發(fā)run()方法內(nèi)部的InterruptedException異常,所以運(yùn)行結(jié)果中有異常輸出,上面說了,當(dāng)觸發(fā)InterruptedException
異常時(shí)候,線程內(nèi)部的中斷標(biāo)志又會(huì)被清除(變?yōu)閒alse),所以在catch中又調(diào)用了this.interrupt();一次,將中斷標(biāo)志置為false
3.run()方法中通過this.isInterrupted()來獲取線程的中斷標(biāo)志,退出循環(huán)(break)
總結(jié)
當(dāng)一個(gè)線程處于被阻塞狀態(tài)或者試圖執(zhí)行一個(gè)阻塞操作時(shí),可以使用 Thread.interrupt()方式中斷該線程,注意此時(shí)將會(huì)拋出一個(gè)InterruptedException的異常,同時(shí)中斷狀態(tài)將會(huì)被復(fù)位(由中斷狀態(tài)改為非中斷狀態(tài))
內(nèi)部有循環(huán)體,可以通過一個(gè)變量來作為一個(gè)信號(hào)控制線程是否中斷,注意變量需要volatile修飾
文中的幾種方式可以結(jié)合起來靈活使用控制線程的中斷
本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關(guān)注服務(wù)器之家的更多內(nèi)容!
原文鏈接:https://itsoku.blog.csdn.net/article/details/100036221