面試突擊:為什么Start方法不能重復調用?而Run方法卻可以?
作者 | 磊哥
來源 | Java面試真題解析(ID:aimianshi666)
轉載請聯系授權(微信ID:GG_Stone)
初學線程時,總是將 run 方法和 start 方法搞混,雖然二者是完全不同的兩個方法,但剛開始使用時很難分清,原因就是因為初次使用時效果貌似是一樣的,如下代碼所示:
public static void main(String[] args) {
// 創建線程一
Thread thread = new Thread(new Runnable() {
@Override
public void run() {
System.out.println("執行線程一");
}
});
// 調用 run 方法
thread.run();
// 創建線程二
Thread thread2 = new Thread(new Runnable() {
@Override
public void run() {
System.out.println("執行線程二");
}
});
// 調用 start 方法
thread2.start();
}
以上程序的執行結果如下:
從上述結果可以看出,二者調用之后的執行效果都是一樣,都可以成功執行任務。但是,如果在執行線程的時候,加上打印當前線程的名稱就能看出二者的不同了,如下代碼所示:
public static void main(String[] args) {
// 創建線程一
Thread thread = new Thread(new Runnable() {
@Override
public void run() {
// 獲取到當前執行線程
Thread currThread = Thread.currentThread();
System.out.println("執行線程一,線程名:" + currThread.getName());
}
});
// 調用 run 方法
thread.run();
// 創建線程二
Thread thread2 = new Thread(new Runnable() {
@Override
public void run() {
// 獲取到當前執行線程
Thread currThread = Thread.currentThread();
System.out.println("執行線程二,線程名:" + currThread.getName());
}
});
// 調用 start 方法
thread2.start();
}
以上程序的執行結果如下:
從上述結果我們可以看出:當調用 run 方法時,其實是調用當前主程序 main
來執行方法體的;而調用 start 方法才是真正的創建一個新線程來執行任務。
區別
1run 方法和 start 方法的第一個區別是:調用 start 方法是真正開啟一個線程來執行任務,而調用 run 方法相當于執行普通方法 run,并不會開啟新線程,如下圖所示:
區別2
run 方法和 start 方法的第二個區別是:run 方法也叫做線程體,它里面包含了具體要執行的業務代碼,當調用 run 方法時,會立即執行 run 方法中的代碼(如果當前線程時間片未用完);而調用 start 方法時,是啟動一個線程并將線程的狀態設置為就緒狀態。也就是說調用 start 方法,并不會立即執行。
區別3
因為 run 方法是普通方法,而普通方法是可以被多次調用的,所以 run 方法可以被調用多次;而 start 方法是創建新線程來執行任務,因為線程只能被創建一次,所以它們的第三個區別是:run 方法可以被調用多次,而 start 方法只能被調用一次。測試代碼如下:
// 創建線程一
Thread thread = new Thread(new Runnable() {
@Override
public void run() {
// 獲取到當前執行的線程
Thread currThread = Thread.currentThread();
System.out.println("執行線程一,線程名:" + currThread.getName());
}
});
// 調用 run 方法
thread.run();
// 多次調用 run 方法
thread.run();
// 創建線程二
Thread thread2 = new Thread(new Runnable() {
@Override
public void run() {
// 獲取到當前執行的線程
Thread currThread = Thread.currentThread();
System.out.println("執行線程二,線程名:" + currThread.getName());
}
});
// 調用 start 方法
thread2.start();
// 多次調用 start 方法
thread2.start();
以上程序的執行結果如下:
從上述結果可以看出,run 方法多次調用可用正常執行,而第二次調用 start
方法時程序就報錯了,提示“IllegalThreadStateException”非法線程狀態異常。
為什么start不能被重復調用?
要找到這個問題的答案,就要查看 start 方法的實現源碼,它的源碼如下:
從 start 源碼實現的第一行,我們就可以得到問題的答案,因為 start 方法在執行時,會先判斷當前線程的狀態是不是等于 0,也就是是否為新建狀態 NEW,如果不等于新建狀態,那么就會拋出“IllegalThreadStateException”非法線程狀態異常,這就是線程的 start 方法不能被重復調用的原因。它的執行過程是:當線程調用了第一個 start 方法之后,線程的狀態就會從新建狀態 NEW,變為就緒狀態 RUNNABLE,此時再次調用 start 方法,JVM 就會判斷出當前的線程已經不等于新建狀態,從而拋出 IllegalThreadStateException 非法線程狀態異常。
總結
run 方法和 start 方法的主要區別如下:
- 方法性質不同:run 是一個普通方法,而 start 是開啟新線程的方法。
- 執行速度不同:調用 run 方法會立即執行任務,調用 start 方法是將線程的狀態改為就緒狀態,不會立即執行。
- 調用次數不同:run 方法可以被重復調用,而 start 方法只能被調用一次。
start 方法之所以不能被重復調用的原因是,線程的狀態是不可逆的,Thread 在 start 的實現源碼中做了判斷,如果線程不是新建狀態 NEW,則會拋出非法線程狀態異常 IllegalThreadStateException。