JAVA 執行緒的幾種狀態

zhaosoft1982發表於2010-03-30

1. 執行緒的幾種狀態

執行緒有四種狀態,任何一個執行緒肯定處於這四種狀態中的一種:

1) 產生(New):執行緒物件已經產生,但尚未被啟動,所以無法執行。如通過new產生了一個執行緒物件後沒對它呼叫start()函式之前。

2) 可執行(Runnable):每個支援多執行緒的系統都有一個排程器,排程器會從執行緒池中選擇一個執行緒並啟動它。當一個執行緒處於可執行狀態時,表示它可能正處於執行緒池中等待排排程器啟動它;也可能它已正在執行。如執行了一個執行緒物件的start()方法後,執行緒就處於可執行狀態,但顯而易見的是此時執行緒不一定正在執行中。

3) 死亡(Dead):當一個執行緒正常結束,它便處於死亡狀態。如一個執行緒的run()函式執行完畢後執行緒就進入死亡狀態。

4) 停滯(Blocked):當一個執行緒處於停滯狀態時,系統排程器就會忽略它,不對它進行排程。當處於停滯狀態的執行緒重新回到可執行狀態時,它有可能重新執行。如通過對一個執行緒呼叫wait()函式後,執行緒就進入停滯狀態,只有當兩次對該執行緒呼叫notify或notifyAll後它才能兩次回到可執行狀態。3. class Object下常用的執行緒函式

wait()、notify()和notifyAll()這三個函式由java.lang.Object類提供,用於協調多個執行緒對共享資料的存取。

3.1 wait()、notify()和notifyAll()

1) wait()函式有兩種形式:第一種形式接受一個毫秒值,用於在指定時間長度內暫停執行緒,使執行緒進入停滯狀態。第二種形式為不帶引數,代表waite()在notify()或notifyAll()之前會持續停滯。

2) 當對一個物件執行notify()時,會從執行緒等待池中移走該任意一個執行緒,並把它放到鎖標誌等待池中;當對一個物件執行notifyAll()時,會從執行緒等待池中移走所有該物件的所有執行緒,並把它們放到鎖標誌等待池中。

3) 當呼叫wait()後,執行緒會釋放掉它所佔有的“鎖標誌”,從而使執行緒所在物件中的其它synchronized資料可被別的執行緒使用。

例17:

下面,我們將對例11中的例子進行修改

class TestThreadMethod extends Thread{

public static int shareVar = 0;

public TestThreadMethod(String name){

super(name);

}

public synchronized void run(){

if(shareVar==0){

for(int i=0; i<10; i++){

shareVar++;

if(shareVar==5){

try{

this.wait(); //(4)

}

catch(InterruptedException e){}

}

}

}

if(shareVar!=0){

System.out.print(Thread.currentThread().getName());

System.out.println(" shareVar = " + shareVar);

this.notify(); //(5)

}

}

}

public class TestThread{

public static void main(String[] args){

TestThreadMethod t1 = new TestThreadMethod("t1");

TestThreadMethod t2 = new TestThreadMethod("t2");

t1.start(); //(1)

//t1.start(); (2)

t2.start(); //(3)

}}

執行結果為:

t2 shareVar = 5

因為t1和t2是兩個不同物件,所以執行緒t2呼叫程式碼(5)不能喚起執行緒t1。如果去掉程式碼(2)的註釋,並註釋掉程式碼(3),結果為:

t1 shareVar = 5

t1 shareVar = 10

這是因為,當程式碼(1)的執行緒執行到程式碼(4)時,它進入停滯狀態,並釋放物件的鎖狀態。接著,程式碼(2)的執行緒執行run(),由於此時 shareVar值為5,所以執行列印語句並呼叫程式碼(5)使程式碼(1)的執行緒進入可執行狀態,然後程式碼(2)的執行緒結束。當程式碼(1)的執行緒重新執行後,它接著執行for()迴圈一直到shareVar=10,然後列印shareVar。

相關文章