1,線程的生命周期
線程從創(chuàng)建、運行到結(jié)束總是處于下面五個狀態(tài)之一:新建狀態(tài)、就緒狀態(tài)、運行狀態(tài)、阻塞狀態(tài)及死亡狀態(tài)。

1.新建狀態(tài)(New):
當用new操作符創(chuàng)建一個線程時, 例如new Thread(r),線程還沒有開始運行,此時線程處在新建狀態(tài)。 當一個線程處于新生狀態(tài)時,程序還沒有開始運行線程中的代碼
2.就緒狀態(tài)(Runnable)
一個新創(chuàng)建的線程并不自動開始運行,要執(zhí)行線程,必須調(diào)用線程的start()方法。當線程對象調(diào)用start()方法即啟動了線程,start()方法創(chuàng)建線程運行的系統(tǒng)資源,并調(diào)度線程運行run()方法。當start()方法返回后,線程就處于就緒狀態(tài)。
處于就緒狀態(tài)的線程并不一定立即運行run()方法,線程還必須同其他線程競爭CPU時間,只有獲得CPU時間才可以運行線程。因為在單CPU的計算機系統(tǒng)中,不可能同時運行多個線程,一個時刻僅有一個線程處于運行狀態(tài)。因此此時可能有多個線程處于就緒狀態(tài)。對多個處于就緒狀態(tài)的線程是由Java運行時系統(tǒng)的線程調(diào)度程序(thread scheduler)來調(diào)度的。
3.運行狀態(tài)(Running)
當線程獲得CPU時間后,它才進入運行狀態(tài),真正開始執(zhí)行run()方法.
4. 阻塞狀態(tài)(Blocked)
線程運行過程中,可能由于各種原因進入阻塞狀態(tài):
1>線程通過調(diào)用sleep方法進入睡眠狀態(tài);
2>線程調(diào)用一個在I/O上被阻塞的操作,即該操作在輸入輸出操作完成之前不會返回到它的調(diào)用者;
3>線程試圖得到一個鎖,而該鎖正被其他線程持有;
4>線程在等待某個觸發(fā)條件;
......
所謂阻塞狀態(tài)是正在運行的線程沒有運行結(jié)束,暫時讓出CPU,這時其他處于就緒狀態(tài)的線程就可以獲得CPU時間,進入運行狀態(tài)。
5. 死亡狀態(tài)(Dead)
有兩個原因會導(dǎo)致線程死亡:
1) run方法正常退出而自然死亡,
2) 一個未捕獲的異常終止了run方法而使線程猝死。
為了確定線程在當前是否存活著(就是要么是可運行的,要么是被阻塞了),需要使用isAlive方法。如果是可運行或被阻塞,這個方法返回true; 如果線程仍舊是new狀態(tài)且不是可運行的, 或者線程死亡了,則返回false.
2, 線程的優(yōu)先級和調(diào)度
Java的每個線程都有一個優(yōu)先級,當有多個線程處于就緒狀態(tài)時,線程調(diào)度程序根據(jù)線程的優(yōu)先級調(diào)度線程運行。
可以用下面方法設(shè)置和返回線程的優(yōu)先級。
· public final void setPriority(int newPriority) 設(shè)置線程的優(yōu)先級。
· public final int getPriority() 返回線程的優(yōu)先級。
newPriority為線程的優(yōu)先級,其取值為1到10之間的整數(shù),也可以使用Thread類定義的常量來設(shè)置線程的優(yōu)先級,這些常量分別為:Thread.MIN_PRIORITY、Thread.NORM_PRIORITY、Thread.MAX_PRIORITY,它們分別對應(yīng)于線程優(yōu)先級的1、5和10,數(shù)值越大優(yōu)先級越高。當創(chuàng)建Java線程時,如果沒有指定它的優(yōu)先級,則它從創(chuàng)建該線程那里繼承優(yōu)先級。
一般來說,只有在當前線程停止或由于某種原因被阻塞,較低優(yōu)先級的線程才有機會運行。
前面說過多個線程可并發(fā)運行,然而實際上并不總是這樣。由于很多計算機都是單CPU的,所以一個時刻只能有一個線程運行,多個線程的并發(fā)運行只是幻覺。在單CPU機器上多個線程的執(zhí)行是按照某種順序執(zhí)行的,這稱為線程的調(diào)度(scheduling)。
大多數(shù)計算機僅有一個CPU,所以線程必須與其他線程共享CPU。多個線程在單個CPU是按照某種順序執(zhí)行的。實際的調(diào)度策略隨系統(tǒng)的不同而不同,通常線程調(diào)度可以采用兩種策略調(diào)度處于就緒狀態(tài)的線程。
(1) 搶占式調(diào)度策略
Java運行時系統(tǒng)的線程調(diào)度算法是搶占式的 (preemptive)。Java運行時系統(tǒng)支持一種簡單的固定優(yōu)先級的調(diào)度算法。如果一個優(yōu)先級比其他任何處于可運行狀態(tài)的線程都高的線程進入就緒狀態(tài),那么運行時系統(tǒng)就會選擇該線程運行。新的優(yōu)先級較高的線程搶占(preempt)了其他線程。但是Java運行時系統(tǒng)并不搶占同優(yōu)先級的線程。換句話說,Java運行時系統(tǒng)不是分時的(time-slice)。然而,基于Java Thread類的實現(xiàn)系統(tǒng)可能是支持分時的,因此編寫代碼時不要依賴分時。當系統(tǒng)中的處于就緒狀態(tài)的線程都具有相同優(yōu)先級時,線程調(diào)度程序采用一種簡單的、非搶占式的輪轉(zhuǎn)的調(diào)度順序。
(2) 時間片輪轉(zhuǎn)調(diào)度策略
有些系統(tǒng)的線程調(diào)度采用時間片輪轉(zhuǎn)(round-robin)調(diào)度策略。這種調(diào)度策略是從所有處于就緒狀態(tài)的線程中選擇優(yōu)先級最高的線程分配一定的CPU時間運行。該時間過后再選擇其他線程運行。只有當線程運行結(jié)束、放棄(yield)CPU或由于某種原因進入阻塞狀態(tài),低優(yōu)先級的線程才有機會執(zhí)行。如果有兩個優(yōu)先級相同的線程都在等待CPU,則調(diào)度程序以輪轉(zhuǎn)的方式選擇運行的線程。
3. 線程狀態(tài)的改變
一個線程在其生命周期中可以從一種狀態(tài)改變到另一種狀態(tài),線程狀態(tài)的變遷如圖所示:

1> 控制線程的啟動和結(jié)束
當一個新建的線程調(diào)用它的start()方法后即進入就緒狀態(tài),處于就緒狀態(tài)的線程被線程調(diào)度程序選中就可以獲得CPU時間,進入運行狀態(tài),該線程就開始運行run()方法。
控制線程的結(jié)束稍微復(fù)雜一點。如果線程的run()方法是一個確定次數(shù)的循環(huán),則循環(huán)結(jié)束后,線程運行就結(jié)束了,線程對象即進入死亡狀態(tài)。如果run()方法是一個不確定循環(huán),早期的方法是調(diào)用線程對象的stop()方法,然而由于該方法可能導(dǎo)致線程死鎖,因此從1.1版開始,不推薦使用該方法結(jié)束線程。一般是通過設(shè)置一個標志變量,在程序中改變標志變量的值實現(xiàn)結(jié)束線程。請看下面的例子:
程序 ThreadStop.java
import java.util.*;
class Timer implements Runnable{
boolean flag=true;
public void run(){
while(flag){
System.out.print("\r\t"+new Date()+"
");
try{
Thread.sleep(1000);
}catch(InterruptedException e){}
}
System.out.println("\n"+Thread.currentThread().getName()+" Stop");
}
public void stopRun(){
flag = false;
}
}
public class ThreadStop{
public static void main(String args[]){
Timer timer = new Timer();
Thread thread = new Thread(timer);
thread.setName("Timer");
thread.start();
for(int i=0;i<100;i++){
System.out.print("\r"+i);
try{
Thread.sleep(100);
}catch(InterruptedException e){}
}
timer.stopRun();
}
}
該程序在Timer類中定義了一個布爾變量flag,同時定義了一個stopRun()方法,在其中將該變量設(shè)置為false。在主程序中通過調(diào)用該方法,從而改變該變量的值,使得run()方法的while循環(huán)條件不滿足,從而實現(xiàn)結(jié)束線程的運行。
說明 在Thread類中除了stop()方法被標注為不推薦(deprecated) 使用外,suspend()方法和resume()方法也被標明不推薦使用,這兩個方法原來用作線程的掛起和恢復(fù).
2> 線程阻塞條件
處于運行狀態(tài)的線程除了可以進入死亡狀態(tài)外,還可能進入就緒狀態(tài)和阻塞狀態(tài)。下面分別討論這兩種情況:
(1) 運行狀態(tài)到就緒狀態(tài)
處于運行狀態(tài)的線程如果調(diào)用了yield()方法,那么它將放棄CPU時間,使當前正在運行的線程進入就緒狀態(tài)。這時有幾種可能的情況:如果沒有其他的線程處于就緒狀態(tài)等待運行,該線程會立即繼續(xù)運行;如果有等待的線程,此時線程回到就緒狀態(tài)狀態(tài)與其他線程競爭CPU時間,當有比該線程優(yōu)先級高的線程時,高優(yōu)先級的線程進入運行狀態(tài),當沒有比該線程優(yōu)先級高的線程時,但有同優(yōu)先級的線程,則由線程調(diào)度程序來決定哪個線程進入運行狀態(tài),因此線程調(diào)用yield()方法只能將CPU時間讓給具有同優(yōu)先級的或高優(yōu)先級的線程而不能讓給低優(yōu)先級的線程。
一般來說,在調(diào)用線程的yield()方法可以使耗時的線程暫停執(zhí)行一段時間,使其他線程有執(zhí)行的機會。
(2) 運行狀態(tài)到阻塞狀態(tài)
有多種原因可使當前運行的線程進入阻塞狀態(tài),進入阻塞狀態(tài)的線程當相應(yīng)的事件結(jié)束或條件滿足時進入就緒狀態(tài)。使線程進入阻塞狀態(tài)可能有多種原因:
① 線程調(diào)用了sleep()方法,線程進入睡眠狀態(tài),此時該線程停止執(zhí)行一段時間。當時間到時該線程回到就緒狀態(tài),與其他線程競爭CPU時間。
Thread類中定義了一個interrupt()方法。一個處于睡眠中的線程若調(diào)用了interrupt()方法,該線程立即結(jié)束睡眠進入就緒狀態(tài)。
② 如果一個線程的運行需要進行I/O操作,比如從鍵盤接收數(shù)據(jù),這時程序可能需要等待用戶的輸入,這時如果該線程一直占用CPU,其他線程就得不到運行。這種情況稱為I/O阻塞。這時該線程就會離開運行狀態(tài)而進入阻塞狀態(tài)。Java語言的所有I/O方法都具有這種行為。
③ 有時要求當前線程的執(zhí)行在另一個線程執(zhí)行結(jié)束后再繼續(xù)執(zhí)行,這時可以調(diào)用join()方法實現(xiàn),join()方法有下面三種格式:
· public void join() throws InterruptedException 使當前線程暫停執(zhí)行,等待調(diào)用該方法的線程結(jié)束后再執(zhí)行當前線程。
· public void join(long millis) throws InterruptedException 最多等待millis毫秒后,當前線程繼續(xù)執(zhí)行。
· public void join(long millis, int nanos) throws InterruptedException 可以指定多少毫秒、多少納秒后繼續(xù)執(zhí)行當前線程。
上述方法使當前線程暫停執(zhí)行,進入阻塞狀態(tài),當調(diào)用線程結(jié)束或指定的時間過后,當前線程線程進入就緒狀態(tài),例如執(zhí)行下面代碼:
t.join();
將使當前線程進入阻塞狀態(tài),當線程t執(zhí)行結(jié)束后,當前線程才能繼續(xù)執(zhí)行。
④ 線程調(diào)用了wait()方法,等待某個條件變量,此時該線程進入阻塞狀態(tài)。直到被通知(調(diào)用了notify()或notifyAll()方法)結(jié)束等待后,線程回到就緒狀態(tài)。
⑤ 另外如果線程不能獲得對象鎖,也進入就緒狀態(tài)。
后兩種情況在下一節(jié)討論。