Java線程調度的一點背景
在各種各樣的線程中,Java虛擬機必須實現一個有優先權的、基於優先級的調度程序。這意味着Java程序中的每一個線程被分配到一定的優先權,使用定義好的范圍內的一個正整數表示。優先級可以被開發者改變。即使線程已經運行了一定時間,Java虛擬機也不會改變其優先級
優先級的值很重要,因為Java虛擬機和下層的操作系統之間的約定是操作系統必須選擇有最高優先權的Java線程運行。所以我們說Java實現了一個基於優先權的調度程序。該調度程序使用一種有優先權的方式實現,這意味着當一個有更高優先權的線程到來時,無論低優先級的線程是否在運行,都會中斷(搶占)它。這個約定對於操作系統來說並不總是這樣,這意味着操作系統有時可能會選擇運行一個更低優先級的線程。(我憎恨多線程的這一點,因為這不能保證任何事情)
理解線程的優先權
接下來,理解線程優先級是多線程學習很重要的一步,尤其是了解yield()函數的工作過程。
- 記住當線程的優先級沒有指定時,所有線程都攜帶普通優先級。
- 優先級可以用從1到10的范圍指定。10表示最高優先級,1表示最低優先級,5是普通優先級。
- 記住優先級最高的線程在執行時被給予優先。但是不能保證線程在啟動時就進入運行狀態。
- 與在線程池中等待運行機會的線程相比,當前正在運行的線程可能總是擁有更高的優先級。
- 由調度程序決定哪一個線程被執行。
- t.setPriority()用來設定線程的優先級。
- 記住在線程開始方法被調用之前,線程的優先級應該被設定。
- 你可以使用常量,如MIN_PRIORITY,MAX_PRIORITY,NORM_PRIORITY來設定優先級
現在,當我們對線程調度和線程優先級有一定理解后,讓我們進入主題。
yield()方法
理論上,yield意味着放手,放棄,投降。一個調用yield()方法的線程告訴虛擬機它樂意讓其他線程占用自己的位置。這表明該線程沒有在做一些緊急的事情。注意,這僅是一個暗示,並不能保證不會產生任何影響。
讓我們列舉一下關於以上定義重要的幾點:
- Yield是一個靜態的原生(native)方法
- Yield告訴當前正在執行的線程把運行機會交給線程池中擁有相同優先級的線程。
- Yield不能保證使得當前正在運行的線程迅速轉換到可運行的狀態
- 它僅能使一個線程從運行狀態轉到可運行狀態,而不是等待或阻塞狀態.
yield()方法使用示例
在下面的示例程序中,我隨意的創建了名為生產者和消費者的兩個線程。生產者設定為最小優先級,消費者設定為最高優先級。在Thread.yield()注釋和非注釋的情況下我將分別運行該程序。沒有調用yield()方法時,雖然輸出有時改變,但是通常消費者行先打印出來,然后事生產者。
調用yield()方法時,兩個線程依次打印,然后將執行機會交給對方,一直這樣進行下去。
join()方法
線程實例的方法join()方法可以使得一個線程在另一個線程結束后再執行。如果join()方法在一個線程實例上調用,當前運行着的線程將阻塞直到這個線程實例完成了執行。
|
1
2
3
|
//Waits for this thread to die.
public
final
void
join()
throws
InterruptedException
|
在join()方法內設定超時,使得join()方法的影響在特定超時后無效。當超時時,主方法和任務線程申請運行的時候是平等的。然而,當涉及sleep時,join()方法依靠操作系統計時,所以你不應該假定join()方法將會等待你指定的時間。
像sleep,join通過拋出InterruptedException對中斷做出回應。
1 package test.core.threads; 2 3 public class JoinExample 4 { 5 public static void main(String[] args) throws InterruptedException 6 { 7 Thread t = new Thread(new Runnable() 8 { 9 public void run() 10 { 11 System.out.println("First task started"); 12 System.out.println("Sleeping for 2 seconds"); 13 try 14 { 15 Thread.sleep(2000); 16 } catch (InterruptedException e) 17 { 18 e.printStackTrace(); 19 } 20 System.out.println("First task completed"); 21 } 22 }); 23 Thread t1 = new Thread(new Runnable() 24 { 25 public void run() 26 { 27 System.out.println("Second task completed"); 28 } 29 }); 30 t.start(); // Line 15 31 t.join(); // Line 16 32 t1.start(); 33 } 34 } 35 36 Output: 37 38 First task started 39 Sleeping for 2 seconds 40 First task completed 41 Second task completed
