Java結束線程的三種方法(愛奇藝面試)


線程屬於一次性消耗品,在執行完run()方法之后線程便會正常結束了,線程結束后便會銷毀,不能再次start,只能重新建立新的線程對象,但有時run()方法是永遠不會結束的。例如在程序中使用線程進行Socket監聽請求,或是其他的需要循環處理的任務。在這種情況下,一般是將這些任務放在一個循環中,如while循環。當需要結束線程時,如何退出線程呢?

有三種方法可以結束線程:

  • 1.設置退出標志,使線程正常退出,也就是當run()方法完成后線程終止,比如使用一個volatile 的變量;

  • 2.使用interrupt()方法中斷線程,包含兩種情況(a)當線程阻塞時,拋出InterruptedException 異常,捕獲這個異常,然后退出,(b)不阻塞情況下,使用函數isInterrupt()函數判斷,如果為true就退出;

  • 3.使用stop方法強行終止線程,使用它們是極端不安全的!不推薦使用,類似於斷電關電腦(不推薦使用,Thread.stop, Thread.suspend, Thread.resume 和Runtime.runFinalizersOnExit 這些終止線程運行的方法已經被廢棄,)

前兩種方法都可以實現線程的正常退出;第3種方法相當於電腦斷電關機一樣,是不安全的方法。

1.使用退出標志終止線程 
一般run()方法執行完,線程就會正常結束,然而,常常有些線程是伺服線程。它們需要長時間的運行,只有在外部某些條件滿足的情況下,才能關閉這些線程。使用一個變量來控制循環,例如:最直接的方法就是設一個boolean類型的標志,並通過設置這個標志為true或false來控制while循環是否退出,代碼示例: 

public class ThreadSafe extends Thread {
    public volatile boolean exit = false; 
        public void run() { 
        while (!exit){
            //do something
        }
    } 
}

 

定義了一個退出標志exit,當exit為true時,while循環退出,exit的默認值為false.在定義exit時,使用了一個Java關鍵字volatile,這個關鍵字的目的是使exit同步,也就是說在同一時刻只能由一個線程來修改exit的值.

2.使用interrupt()方法中斷當前線程 
使用interrupt()方法來中斷線程有兩種情況:

1.線程處於阻塞狀態,如使用了sleep,同步鎖的wait,socket中的receiver,accept等方法時,會使線程處於阻塞狀態。當調用線程的interrupt()方法時,會拋出InterruptException異常。阻塞中的那個方法拋出這個異常,通過代碼捕獲該異常,然后break跳出循環狀態,從而讓我們有機會結束這個線程的執行。通常很多人認為只要調用interrupt方法線程就會結束,實際上是錯的, 一定要先捕獲InterruptedException異常之后通過break來跳出循環,才能正常結束run方法。  

public class ThreadSafe extends Thread {
    public void run() { 
        while (true){
            try{
                    Thread.sleep(5*1000);//阻塞5妙
                }catch(InterruptedException e){
                    e.printStackTrace();
                    break;//捕獲到異常之后,執行break跳出循環。
                }
        }
    } 
}

 

2.線程未處於阻塞狀態,使用isInterrupted()判斷線程的中斷標志來退出循環。當使用interrupt()方法時,中斷標志就會置true,和使用自定義的標志來控制循環是一樣的道理。 

 

public class ThreadSafe extends Thread {
    public void run() { 
        while (!isInterrupted()){
            //do something, but no throw InterruptedException
        }
    } 
}

 

為什么要區分進入阻塞狀態和和非阻塞狀態兩種情況了,是因為當阻塞狀態時,如果有interrupt()發生,系統除了會拋出InterruptedException異常外,還會調用interrupted()函數,調用時能獲取到中斷狀態是true的狀態,調用完之后會復位中斷狀態為false,所以異常拋出之后通過isInterrupted()是獲取不到中斷狀態是true的狀態,從而不能退出循環,因此在線程未進入阻塞的代碼段時是可以通過isInterrupted()來判斷中斷是否發生來控制循環,在進入阻塞狀態后要通過捕獲異常來退出循環。因此使用interrupt()來退出線程的最好的方式應該是兩種情況都要考慮:也就是說如果阻塞了,調用isInterrupted()函數,會返回false,而不是預期的true。不阻塞的話,isInterrupted()函數返回true; 

public class ThreadSafe extends Thread {
    public void run() { 
        while (!isInterrupted()){ //非阻塞過程中通過判斷中斷標志來退出
            try{
                Thread.sleep(5*1000);//阻塞過程捕獲中斷異常來退出
            }catch(InterruptedException e){
                e.printStackTrace();
                break;//捕獲到異常之后,執行break跳出循環。
            }
        }
    } 
}

3.使用stop方法終止線程 

程序中可以直接使用thread.stop()來強行終止線程,但是stop方法是很危險的,就象突然關閉計算機電源,而不是按正常程序關機一樣,可能會產生不可預料的結果,不安全主要是:thread.stop()調用之后,創建子線程的線程就會拋出ThreadDeatherror的錯誤,並且會釋放子線程所持有的所有鎖。一般任何進行加鎖的代碼塊,都是為了保護數據的一致性,如果在調用thread.stop()后導致了該線程所持有的所有鎖的突然釋放(不可控制),那么被保護數據就有可能呈現不一致性,其他線程在使用這些被破壞的數據時,有可能導致一些很奇怪的應用程序錯誤。因此,並不推薦使用stop方法來終止線程。

線程池的銷毀,其實也是兩種情況

(a)正常停止使用shutdown(),設置線程池狀態位SHUTDOWN,函數調用 interrupt();不會立即終止線程池,而是要等所有任務緩存隊列中的任務都執行完后才終止,但再也不會接受新的任務

(b)強行停止,使用shutdownNow()函數,設置線程池狀態為STOP,也是調用了interrupt(),不安全的函數,立即終止線程池,並嘗試打斷正在執行的任務,並且清空任務緩存隊列,返回尚未執行的任務

具體的可以參考這個:深入理解Java之線程池(愛奇藝面試)

參考:Java結束線程的三種方法


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM