Java中如何結束執行緒
阿新 • • 發佈:2019-01-02
一般來說執行緒執行完run()之後就自動結束了,不過有些時候我們需要執行緒不停的做一些事情,也就是使用while迴圈,那麼這時候該如何停止執行緒呢?
這個問題需要分情況來討論,如果執行緒做的事情不是耗時的,那麼只需要使用一個標誌即可,具體的程式碼如下:
如果執行緒做的事情是耗時或者說阻塞的(如呼叫了sleep,同步鎖的wait,socket的receiver,accept等方法),那麼就需要用到interrupt()了,呼叫該函式時會丟擲InterruptedException異常,程式碼中通過捕獲該異常,然後break出迴圈,就可以了。程式碼如下:
public static boolean interrupted()
測試當前執行緒是否已經中斷,並且清除執行緒的中斷狀態。
public boolean isInterrupted()
測試執行緒是否已經中斷,執行緒的中斷狀態不受該方法的影響。
這兩個函式的返回值含義:
如果當前執行緒已經中斷,則返回 true;否則返回 false。
其實當執行緒阻塞並且呼叫了interrupt()時,不止是丟擲InterruptedException異常還會呼叫interrupted()來清除執行緒的中斷狀態,所以在catch裡面呼叫isInterrupted()會返回false。
為了更好的理解這兩個函式,我們看個例子:
...
1300 in thread isInterrupted() false
exec interrupt
in timer isInterrupted() true
1301 in thread isInterrupted() true
in thread after interrupted() true
java.lang.InterruptedException: sleep interrupted
at java.lang.Thread.sleep(Native Method)
at com.example.wd.MyThread.run(Tes1.java:21)
還可能是這樣:
...
1568 in thread isInterrupted() false
exec interrupt
in thread after interrupted() true
in timer isInterrupted() true
java.lang.InterruptedException: sleep interrupted
at java.lang.Thread.sleep(Native Method)
at com.example.wd.MyThread.run(Tes1.java:21)
由於在sleep時執行了interrupt()所以會丟擲異常。
如果把if(isInterrupted()==true)改成if(Thread.interrupted()==true)那麼列印的結果就不同了,如下:
...
1582 in thread isInterrupted() false
exec interrupt
in timer isInterrupted() true
in thread after interrupted() false
還可能是這樣:
...
1771 in thread isInterrupted() false
exec interrupt
in timer isInterrupted() true
1772 in thread isInterrupted() false
in thread after interrupted() false
也可能是這樣:
...
476 in thread isInterrupted() false
in thread after interrupted() false
in timer isInterrupted() false
雖然執行了interrupt(),但是由於Thread.interrupted()可以清除中斷,所以並不會丟擲異常。
前面我們針對執行緒是否耗時,給出了停止執行緒的方法,其實還有一個辦法那就是使用thread.stop()來強行終止執行緒,不過這個由於該方法不安全已經廢棄掉了,因為他有下面兩個缺陷:
1. 立即丟擲ThreadDeath異常,線上程的run()方法內,任何一點都有可能丟擲ThreadDeath異常,包括在catch或finally語句中。
2. 釋放該執行緒所持有的所有的鎖。
好了,關於執行緒如何停止的問題到這裡就圓滿收官了,大家有什麼問題歡迎留言討論。
這個問題需要分情況來討論,如果執行緒做的事情不是耗時的,那麼只需要使用一個標誌即可,具體的程式碼如下:
- class MyThread extends Thread {
- privatevolatileboolean isStop = false;
- publicvoid run() {
- while (!isStop) {
-
System.out.println("do something"
- }
- }
- publicvoid setStop() {
- isStop = true;
- }
- }
如果執行緒做的事情是耗時或者說阻塞的(如呼叫了sleep,同步鎖的wait,socket的receiver,accept等方法),那麼就需要用到interrupt()了,呼叫該函式時會丟擲InterruptedException異常,程式碼中通過捕獲該異常,然後break出迴圈,就可以了。程式碼如下:
- class MyThread extends Thread {
- privatevolatileboolean isStop = false;
- publicvoid run() {
- while (!isStop) {
- try {
- System.out.println("do something");
- Thread.sleep(1000);
- } catch (InterruptedException e) {
-
e.printStackTrace();
- break;//如果沒有這一句,而且不呼叫setStop(),執行緒並不會結束,需要特別注意一下
- }
- }
- }
- publicvoid setStop() {
- isStop = true;
- }
- }
- publicclass Tes1 {
- publicstaticvoid main(String[] args) {
- MyThread t1 = new MyThread();
- t1.start();
- try {
- Thread.sleep(10);
- t1.setStop();
- t1.interrupt();
- } catch (InterruptedException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- }
public static boolean interrupted()
測試當前執行緒是否已經中斷,並且清除執行緒的中斷狀態。
public boolean isInterrupted()
測試執行緒是否已經中斷,執行緒的中斷狀態不受該方法的影響。
這兩個函式的返回值含義:
如果當前執行緒已經中斷,則返回 true;否則返回 false。
其實當執行緒阻塞並且呼叫了interrupt()時,不止是丟擲InterruptedException異常還會呼叫interrupted()來清除執行緒的中斷狀態,所以在catch裡面呼叫isInterrupted()會返回false。
為了更好的理解這兩個函式,我們看個例子:
- class MyThread extends Thread {
- privateint counter = 0;
- publicvoid run() {
- boolean done = false;
- try{
- Thread.sleep(10);
- }catch(InterruptedException ie){
- ie.printStackTrace();
- return;
- }
- while (counter < 2000 &&!done) {
- System.out.println(counter++ + " in thread isInterrupted() "+isInterrupted());
- if(isInterrupted()==true){
- try{
- System.out.println("in thread after interrupted() "+isInterrupted());
- sleep(100);
- break;
- }catch(InterruptedException ie){
- ie.printStackTrace();
- break;
- }
- }
- }
- }
- }
- publicclass Tes1 {
- publicstaticvoid main(String[] args) {
- final MyThread t1 = new MyThread();
- t1.start();
- new Timer(true).schedule(new TimerTask() {
- publicvoid run() {
- System.out.println("exec interrupt");
- t1.interrupt();
- System.out.println("in timer isInterrupted() "+t1.isInterrupted());
- }
- }, 20);
- }
- }
...
1300 in thread isInterrupted() false
exec interrupt
in timer isInterrupted() true
1301 in thread isInterrupted() true
in thread after interrupted() true
java.lang.InterruptedException: sleep interrupted
at java.lang.Thread.sleep(Native Method)
at com.example.wd.MyThread.run(Tes1.java:21)
還可能是這樣:
...
1568 in thread isInterrupted() false
exec interrupt
in thread after interrupted() true
in timer isInterrupted() true
java.lang.InterruptedException: sleep interrupted
at java.lang.Thread.sleep(Native Method)
at com.example.wd.MyThread.run(Tes1.java:21)
由於在sleep時執行了interrupt()所以會丟擲異常。
如果把if(isInterrupted()==true)改成if(Thread.interrupted()==true)那麼列印的結果就不同了,如下:
...
1582 in thread isInterrupted() false
exec interrupt
in timer isInterrupted() true
in thread after interrupted() false
還可能是這樣:
...
1771 in thread isInterrupted() false
exec interrupt
in timer isInterrupted() true
1772 in thread isInterrupted() false
in thread after interrupted() false
也可能是這樣:
...
476 in thread isInterrupted() false
in thread after interrupted() false
in timer isInterrupted() false
雖然執行了interrupt(),但是由於Thread.interrupted()可以清除中斷,所以並不會丟擲異常。
前面我們針對執行緒是否耗時,給出了停止執行緒的方法,其實還有一個辦法那就是使用thread.stop()來強行終止執行緒,不過這個由於該方法不安全已經廢棄掉了,因為他有下面兩個缺陷:
1. 立即丟擲ThreadDeath異常,線上程的run()方法內,任何一點都有可能丟擲ThreadDeath異常,包括在catch或finally語句中。
2. 釋放該執行緒所持有的所有的鎖。
好了,關於執行緒如何停止的問題到這裡就圓滿收官了,大家有什麼問題歡迎留言討論。