Java多線程程式中經常用到的方法有以下幾個:run(),start(),wait(),sleep(),notify(),notifyAll(),yield(),join(),還有一個重要的關鍵字 synchronized。下面分別對這些方法進行解釋:
一、run()和start()
這兩個方法應該都比較熟悉,把需要平行處理的代碼放在run()方法中,start()方法啟動線程將自動調用 run()方法,這是由Java的記憶體機制規定的。並且run()方法必須是public存取權限,傳回值類型為void。
二、關鍵字Synchronized
這個關鍵字用於保護共用資料,當然前提是要分清哪些資料是共用資料。每個對象都有一個鎖標誌,當一個線程訪問該對象時,被Synchronized修飾的資料將被“上鎖”,阻止其他線程訪問。當前線程訪問完這部分資料後釋放鎖標誌,其他線程就可以訪問了。
public ThreadTest implements Runnable
{
public synchronized void run(){
for(int i=0;i<10;i++)
{
System.out.println(" " + i);
}
}
public static void main(String[] args)
{
Runnable r1 = new ThreadTest();
Runnable r2 = new ThreadTest();
Thread t1 = new Thread(r1);
Thread t2 = new Thread(r2);
t1.start();
t2.start();
}
}
以上這段程式中的 i 變數並不是共用資料,也就是這裡的Synchronized關鍵字並未起作用。因為t1,t2兩個線程是兩個對象(r1,r2)的線程。不同的對象其資料是不同的,所以r1和r2兩個對象的i變數是並不是共用資料。
當把代碼改成如下:Synchronized關鍵字才會起作用
Runnable r = new ThreadTest();
Thread t1 = new Thread(r);
Thread t2 = new Thread(r);
t1.start();
t2.start();
三、sleep()
使當前線程(即調用該方法的線程)暫停執行一段時間,讓其他線程有機會繼續執行,但它並不釋放對象鎖。也就是如果有Synchronized同步塊,其他線程仍然不同訪問共用資料。注意該方法要捕獲異常
比如有兩個線程同時執行(沒有Synchronized),一個線程優先順序為MAX_PRIORITY,另一個為MIN_PRIORITY,如果沒有Sleep()方法,只有高優先順序的線程執行完成後,低優先順序的線程才能執行;但當高優先順序的線程sleep(5000)後,低優先順序就有機會執行了。
總之,sleep()可以使低優先順序的線程得到執行的機會,當然也可以讓同優先順序、高優先順序的線程有執行的機會。
四、join()
join()方法使調用該方法的線程在此之前執行完畢,也就是等待調用該方法的線程執行完畢後再往下繼續執行。注意該方法也要捕獲異常。
五、yield()
它與sleep()類似,只是不能由使用者指定暫停多長時間,並且yield()方法只能讓同優先順序的線程有執行的機會。
六、wait()和notify()、notifyAll()
這三個方法用於協調多個線程對共用資料的存取,所以必須在Synchronized語句塊內使用這三個方法。前面說過Synchronized這個關鍵字用於保護共用資料,阻止其他線程對共用資料的存取。但是這樣程式的流程就很不靈活了,如何才能在當前線程還沒退出Synchronized資料區塊時讓其他線程也有機會訪問共用資料呢?此時就用這三個方法來靈活控制。
wait()方法使當前線程暫停執行並釋放對象鎖標誌,讓其他線程可以進入Synchronized資料區塊,當前線程被放入對象等待池中。當調用 notify()方法後,將從對象的等待池中移走一個任意的線程並放到鎖標誌等待池中,只有
鎖標誌等待池中的線程能夠擷取鎖標誌;如果鎖標誌等待池中沒有線程,則notify()不起作用。
notifyAll()則從對象等待池中移走所有等待那個對象的線程並放到鎖標誌等待池中。
注意 這三個方法都是java.lang.Ojbect的方法!