一run()和start()
這兩個方法應該都比較熟悉把需要並行處理的代碼放在run()方法中start()方法啟動線程將自動調用 run()方法這是由Java的內存機制規定的並且run()方法必須是public訪問權限返回值類型為void
二關鍵字Synchronized
這個關鍵字用於保護共享數據當然前提是要分清哪些數據是共享數據每個對象都有一個鎖標志當一個線程訪問該對象時被Synchronized修飾的數據將被上鎖阻止其他線程訪問當前線程訪問完這部分數據後釋放鎖標志其他線程就可以訪問了
public ThreadTest implements Runnable
{
public synchronized void run(){
for(int i=;i<;i++)
{
Systemoutprintln( + i);
}
}
public static void main(String[] args)
{
Runnable r = new ThreadTest();
Runnable r = new ThreadTest();
Thread t = new Thread(r);
Thread t = new Thread(r);
tstart();
tstart();
}
}
以上這段程序中的 i 變量並不是共享數據也就是這裡的Synchronized關鍵字並未起作用因為tt兩個線程是兩個對象(rr)的線程不同的對象其數據是不同的所以r和r兩個對象的i變量是並不是共享數據
當把代碼改成如下Synchronized關鍵字才會起作用
Runnable r = new ThreadTest();
Thread t = new Thread(r);
Thread t = new Thread(r);
tstart();
tstart();
三sleep()
使當前線程(即調用該方法的線程)暫停執行一段時間讓其他線程有機會繼續執行但它並不釋放對象鎖也就是如果有Synchronized同步塊其他線程仍然不同訪問共享數據注意該方法要捕獲異常
比如有兩個線程同時執行(沒有Synchronized)一個線程優先級為MAX_PRIORITY另一個為MIN_PRIORITY如果沒有Sleep()方法只有高優先級的線程執行完成後低優先級的線程才能執行但當高優先級的線程sleep()後低優先級就有機會執行了
總之sleep()可以使低優先級的線程得到執行的機會當然也可以讓同優先級高優先級的線程有執行的機會
四join()
join()方法使調用該方法的線程在此之前執行完畢也就是等待調用該方法的線程執行完畢後再往下繼續執行注意該方法也要捕獲異常
五yield()
它與sleep()類似只是不能由用戶指定暫停多長時間並且yield()方法只能讓同優先級的線程有執行的機會
六wait()和notify()notifyAll()
這三個方法用於協調多個線程對共享數據的存取所以必須在Synchronized語句塊內使用這三個方法前面說過Synchronized這個關鍵字用於保護共享數據阻止其他線程對共享數據的存取但是這樣程序的流程就很不靈活了如何才能在當前線程還沒退出Synchronized數據塊時讓其他線程也有機會訪問共享數據呢?此時就用這三個方法來靈活控制
wait()方法使當前線程暫停執行並釋放對象鎖標志讓其他線程可以進入Synchronized數據塊當前線程被放入對象等待池中當調用 notify()方法後將從對象的等待池中移走一個任意的線程並放到鎖標志等待池中只有鎖標志等待池中的線程能夠獲取鎖標志如果鎖標志等待池中沒有線程則notify()不起作用
notifyAll()則從對象等待池中移走所有等待那個對象的線程並放到鎖標志等待池中
From:http://tw.wingwit.com/Article/program/Java/gj/201311/27686.html