概念
1.synchronized 關鍵字,代表這個方法加鎖,相當於不管哪一個線程(例如線程A),
運行到這個方法時,都要檢查有沒有其它線程B(或者C、 D等)正在用這個方法(或者該類
的其他同步方法),有的話要等正在使用synchronized方法的線程B(或者C 、D)運行完
這個方法後再運行此線程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