Java_synchronized線程鎖解釋

synchronized

概念

1.synchronized 關鍵字,代表這個方法加鎖,相當於不管哪一個線程(例如線程A),
運行到這個方法時,都要檢查有沒有其它線程B(或者CD等)正在用這個方法(或者該類
的其他同步方法),有的話要等正在使用synchronized方法的線程B(或者CD)運行完
這個方法後再運行此線程A,沒有的話,鎖定調用者,然後直接運行。它包括兩種用法:
synchronized 方法和 synchronized 塊。
2.Java語言的關鍵字,可用來給對象和方法或者代碼塊加鎖,當它鎖定一個方法或者一個
代碼塊的時候,同一時刻最多隻有一個線程執行這段代碼。當兩個併發線程訪問同一個
對象object中的這個加鎖同步代碼塊時,一個時間內只能有一個線程得到執行。另一個
線程必須等待當前線程執行完這個代碼塊以後才能執行該代碼塊。然而,當一個線程訪問
object的一個加鎖代碼塊時,另一個線程仍可以訪問該object中的非加鎖代碼塊。

方法

1.方法聲明時使用,放在範圍操作符(public)之後,返回類型聲明(void)之前.這時,
線程獲得的是成員鎖,即一次只能有一個線程進入該方法,其他線程要想在此時調用該方法,
只能排隊等候,當前線程(就是在synchronized方法內部的線程)執行完該方法後,別的線程
才能進入.

public synchronized void synMethod(){
//方法體
}

如在線程t1中有語句obj.synMethod(); 那麼由於synMethod被synchronized修飾,
在執行該語句前, 需要先獲得調用者obj的對象鎖, 如果其他線程(如t2)已經鎖定了obj 
(可能是通過obj.synMethod,也可能是通過其他被synchronized修飾的方法
obj.otherSynMethod鎖定的obj), t1需要等待直到其他線程(t2)釋放obj, 然後t1鎖定
obj, 執行synMethod方法. 返回之前釋放obj鎖.
2.對某一代碼塊使用,synchronized後跟括號,括號裏是變量,這樣,一次只有一個線程進入
該代碼塊.此時,線程獲得的是成員鎖.

public Object synMethod(Object a1){
    synchronized(a1){
//一次只能有一個線程進入
  }
}
3.synchronized後面括號裏是一對象,此時,線程獲得的是對象鎖.
public classMyThread implements Runnable{
    public static void main(Stringargs[]){
      MyThread mt=new MyThread();
        Thread t1=newThread(mt,"t1");
        Thread t2=newThread(mt,"t2");
        Thread t3=newThread(mt,"t3");
        Thread t4=newThread(mt,"t4");
        Thread t5=newThread(mt,"t5");
        Thread t6=newThread(mt,"t6");
        t1.start();
        t2.start();
        t3.start();
        t4.start();
        t5.start();
        t6.start();
}   
public void run(){
    synchronized(this){
      System.out.println(Thread.currentThread().getName());
   }
}
對於3,如果線程進入,則得到當前對象鎖,那麼別的線程在該類所有對象上的任何操作都不
能進行.在對象級使用鎖通常是一種比較粗糙的方法。爲什麼要將整個對象都上鎖,而不
允許其他線程短暫地使用對象中其他同步方法來訪問共享資源?如果一個對象擁有多個
資源,就不需要只爲了讓一個線程使用其中一部分資源,就將所有線程都鎖在外面。由於
每個對象都有鎖,可以如下所示使用虛擬對象來上鎖:
class FineGrainLock{
 MyMemberClassx,y;
 Object xlock = new Object(), ylock = newObject();
 public void foo(){
     synchronized(xlock){
     //accessxhere
      }
     //dosomethinghere-butdon'tusesharedresources
      synchronized(ylock){
      //accessyhere
      }
 }
    public void bar(){
      synchronized(this){
         //accessbothxandyhere
     }
    //dosomethinghere-butdon'tusesharedresources
    }
}
4.synchronized後面括號裏是類,此時,線程獲得的是對象鎖.
classArrayWithLockOrder{
 private static long num_locks=0;
 private long lock_order;
 private int[] arr;
 public ArrayWithLockOrder(int[]a){
     arr=a;
      synchronized(ArrayWithLockOrder.class){//-----這裏
          num_locks++;//鎖數加1。
           lock_order=num_locks;//爲此對象實例設置唯一的lock_order。
      }
 }
 publiclonglockOrder(){
      returnlock_order;
 }
 public int[] array(){
     return arr;
 }
}
class SomeClass implements Runnable{
 public int sumArrays(ArrayWithLockOrdera1,ArrayWithLockOrdera2){
      intvalue=0;
      ArrayWithLockOrderfirst=a1;//保留數組引用的一個
     ArrayWithLockOrderlast=a2;//本地副本。
     intsize=a1.array().length;
      if(size==a2.array().length){
         if(a1.lockOrder()>a2.lockOrder()){//確定並設置對象的鎖定順序。
             first=a2;
            last=a1;
           }
         synchronized(first){//按正確的順序鎖定對象。
             synchronized(last){
                 int[]arr1=a1.array();
                 int[]arr2=a2.array();
                 for(int i=0;i<size;i++)
                       value+=arr1[i]+arr2[i];
               }
          }
     }
 return value;
 }
 public void run(){
 //
 }
}
對於4,如果線程進入,則線程在該類中所有操作不能進行,包括靜態變量和靜態方法,實際
上,對於含有靜態方法和靜態變量的代碼塊的同步,我們通常用4來加鎖.

理解

一、當兩個併發線程訪問同一個對象object中的這個synchronized(this)同步代碼塊時,
一個時間內只能有一個線程得到執行。另一個線程必須等待當前線程執行完這個代碼塊
以後才能執行該代碼塊。
二、然而,另一個線程仍然可以訪問該object中的非synchronized(this)同步代碼塊。
三、尤其關鍵的是,當一個線程訪問object的一個synchronized(this)同步代碼塊時,
其他線程對object中所有其它synchronized(this)同步代碼塊的訪問將被阻塞。
四、第三個例子同樣適用其它同步代碼塊,它就獲得了這個object的對象鎖。結果,其它
線程對該object對象所有同步代碼部分的訪問都被暫時阻塞。
五、以上規則對其它對象鎖同樣適用。

特別注意:
僅供參考學習,轉載請附上原文鏈接
分享學習心得,如有侵權,望聯繫本人處理
還在讀大學的程序員,項目經驗少,如有紕漏,感謝指正
需要源代碼請聯繫本人
謝謝配合

如果這篇文章對您有幫助,小小的點個贊,算是給小學弟的鼓勵吧!謝謝大佬!!/呱呱.jpg

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章