線程對象屬於一次性消耗品,通常線程執行完run方法以後,線程就正常結束了,線程結束以後就報廢了,不能再次start,只能新建一個線程對象。但有時run方法是永遠不會結束的。例如在程序中使用線程進行Socket監聽請求,或是其餘的須要循環處理的任務。在這種狀況下,通常是將這些任務放在一個循環中,如while循環。當須要結束線程時,如何退出線程呢?安全
有三種方法能夠結束線程:socket
1. 使用退出標誌,使線程正常退出,也就是當run方法完成後線程終止函數
2. 使用interrupt()方法中斷線程spa
3. 使用stop方法強行終止線程(不推薦使用,可能發生不可預料的結果)線程
前兩種方法均可以實現線程的正常退出,也就是要談的優雅結束線程;第3種方法至關於電腦斷電關機同樣,是不安全的方法。code
一、使用退出標誌終止線程對象
使用一個變量來控制循環,例如最直接的方法就是設一個boolean類型的標誌,並經過設置這個標誌爲true或false來控制while循環是否退出。代碼以下:blog
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的值,get
二、使用interrupt()方法終止線程同步
使用interrupt()方法來終端線程可分爲兩種狀況:
線程處於阻塞狀態,如使用了sleep,同步鎖的wait,socket的receiver,accept等方法時,會使線程處於阻塞狀態。當調用線程的interrupt()方法時,系統會拋出一個InterruptedException異常,代碼中經過捕獲異常,而後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跳出循環。 } } } }
線程未進入阻塞狀態,使用isInterrupted()判斷線程的中斷標誌來退出循環,當使用interrupt()方法時,中斷標誌就會置true,和使用自定義的標誌來控制循環是同樣的道理。
public class ThreadSafe extends Thread { public void run() { while (!isInterrupted()){ //do something, but no tthrow InterruptedException } } }
爲何要區分進入阻塞狀態和和非阻塞狀態兩種狀況了,是由於當阻塞狀態時,若是有interrupt()發生,系統除了會拋出InterruptedException異常外,還會調用interrupted()函數,調用時能獲取到中斷狀態是true的狀態,調用完以後會復位中斷狀態爲false,因此異常拋出以後經過isInterrupted()是獲取不到中斷狀態是true的狀態,從而不能退出循環,所以在線程未進入阻塞的代碼段時是能夠經過isInterrupted()來判斷中斷是否發生來控制循環,在進入阻塞狀態後要經過捕獲異常來退出循環。所以使用interrupt()來退出線程的最好的方式應該是兩種狀況都要考慮:
public class ThreadSafe extends Thread { public void run() { while (!isInterrupted()){ //非阻塞過程當中經過判斷中斷標誌來退出 try{ Thread.sleep(5*1000);//阻塞過程捕獲中斷異常來退出 }catch(InterruptedException e){ e.printStackTrace(); break;//捕獲到異常以後,執行break跳出循環。 } } } }
三、使用stop方法終止線程
程序中能夠直接使用thread.stop()來強行終止線程,可是stop方法是很危險的,就象忽然關閉計算機電源,而不是按正常程序關機同樣,可能會產生不可預料的結果,不安全主要是:thread.stop()調用以後,建立子線程的線程就會拋出ThreadDeatherror的錯誤,而且會釋放子線程所持有的全部鎖。通常任何進行加鎖的代碼塊,都是爲了保護數據的一致性,若是在調用thread.stop()後致使了該線程所持有的全部鎖的忽然釋放(不可控制),那麼被保護數據就有可能呈現不一致性,其餘線程在使用這些被破壞的數據時,有可能致使一些很奇怪的應用程序錯誤。所以,並不推薦使用stop方法來終止線程。