>synchronized
synchronized 關鍵字,它包括兩種用法:synchronized 方法和 synchronized 區塊。
1>synchronized 方法:透過在方法宣告中加入 synchronized關鍵字來宣告 synchronized 方法。
public synchronized void accessVal(int newVal);
synchronized 方法控制對類別成員的存取權:每個類別鎖定程式的鎖定,每個實例可執行一個對類別成員的存取:每個類別鎖定變數的存取權執行,否則所屬執行緒阻塞,方法一旦執行,就獨佔該鎖,直到從該方法返回時才將鎖釋放,此後被阻塞的執行緒方能獲得該鎖,重新進入可執行狀態。這種機制確保了同一時刻對於每一個類別實例,其所有宣告為synchronized 的成員函數中至多只有一個處於可執行狀態(因為至多只有一個能夠獲得該類別實例對應的鎖),從而有效避免了類別成員變數的存取衝突(只要所有可能存取類別成員變數的方法都被宣告為synchronized)
在Java 中,且不光是類別實例,每個類別也對應一把鎖,這樣我們也可將類別的靜態成員函數聲明為synchronized ,以控制其對類別的靜態成員變數的存取。 synchronized 方法的缺陷:若將一個大的方法宣告為synchronized 將會大大影響效率,典型地,若將執行緒類別的方法run() 宣告為synchronized ,由於在執行緒的整個生命期內它一直在運行,因此將導致它對本類任何synchronized 方法的呼叫都永遠不會成功。當然我們可以透過將存取類別成員變數的程式碼放到專門的方法中,將其宣告為synchronized ,並在主方法中呼叫來解決這一問題,但是Java 為我們提供了更好的解決辦法,那就是synchronized 塊。
2> synchronized 區塊:透過 synchronized關鍵字來宣告synchronized 區塊。文法如下:
synchronized(syncObject) {
ized 區塊是這樣一個程式碼區塊,其中的程式碼必須取得物件syncObject (如前所述,可以是類別實例或類別)的鎖方能執行,具體機制同前所述。由於可以針對任意程式碼區塊,且可任意指定上鎖的對象,故靈活性較高。
對synchronized(this)的一些理解 :
一、當兩個並發線程訪問同一個物件object中的這個synchronized(this)同步程式碼區塊時,一個時間內只能有一個執行緒來執行。另一個執行緒必須等待目前執行緒執行完這個程式碼區塊以後才能執行該程式碼區塊。
二、然而,當一個執行緒存取object的一個synchronized(this)同步程式碼區塊時,另一個執行緒仍然可以存取該object中的非synchronized(this)同步程式碼區塊。
三、尤其關鍵的是,當一個執行緒存取object的一個synchronized(this)同步程式碼區塊時,其他執行緒對object中所有其它synchronized(this)同步程式碼區塊的存取將被阻塞。