首先得明白Java中synchronized用法
synchronized
- 在修飾代碼塊的時候需要一個reference對象作為鎖的對象.
- 在修飾方法的時候默認是當(dāng)前對象作為鎖的對象.
- 在修飾類時候默認是當(dāng)前類的Class對象作為鎖的對象.
下面具體講講三種鎖得區(qū)別與用法
方法鎖(synchronized修飾方法時)
通過在方法聲明中加入 synchronized關(guān)鍵字來聲明 synchronized 方法。
synchronized 方法控制對類成員變量的訪問:
每個類實例對應(yīng)一把鎖囚聚,每個 synchronized 方法都必須獲得調(diào)用該方法的類實例的鎖方能執(zhí)行援奢,否則所屬線程阻塞,方法一旦執(zhí)行朋蔫,就獨占該鎖,直到從該方法返回時才將鎖釋放却汉,此后被阻塞的線程方能獲得該鎖,重新進入可執(zhí)行狀態(tài)荷并。這種機制確保了同一時刻對于每一個類實例合砂,其所有聲明為 synchronized 的成員函數(shù)中至多只有一個處于可執(zhí)行狀態(tài),從而有效避免了類成員變量的訪問沖突源织。
對象鎖(synchronized修飾方法或代碼塊)
當(dāng)一個對象中有synchronized method或synchronized block的時候調(diào)用此對象的同步方法或進入其同步區(qū)域時翩伪,就必須先獲得對象鎖。如果此對象的對象鎖已被其他調(diào)用者占用谈息,則需要等待此鎖被釋放缘屹。(方法鎖也是對象鎖中的一種)
java的所有對象都含有1個互斥鎖,這個鎖由JVM自動獲取和釋放侠仇。線程進入synchronized方法的時候獲取該對象的鎖轻姿,當(dāng)然如果已經(jīng)有線程獲取了這個對象的鎖,那么當(dāng)前線程會等待逻炊;synchronized方法正常返回或者拋異常而終止互亮,JVM會自動釋放對象鎖。這里也體現(xiàn)了用synchronized來加鎖的1個好處余素,方法拋異常的時候豹休,鎖仍然可以由JVM來自動釋放。
對象鎖的兩種形式:
public class TestSynchronized1{
//方法鎖(實質(zhì)也是對象鎖的一種桨吊,鎖定的對象是this)
public synchornized void method1(){
System.out.println("This is a test");
}
//對象鎖
public void method2(){
synchornized(this){
System.out.println("This is a test");
}
}
}
類鎖(synchronized 修飾靜態(tài)的方法或代碼塊)
由于一個class不論被實例化多少次威根,其中的靜態(tài)方法和靜態(tài)變量在內(nèi)存中都只有一份凤巨。所以,一旦一個靜態(tài)的方法被申明為synchronized洛搀。此類所有的實例化對象在調(diào)用此方法敢茁,共用同一把鎖,我們稱之為類鎖姥卢。
對象鎖是用來控制實例方法之間的同步卷要,類鎖是用來控制靜態(tài)方法(或靜態(tài)變量互斥體)之間的同步。
類鎖只是一個概念上的東西独榴,并不是真實存在的僧叉,它只是用來幫助我們理解鎖定實例方法和靜態(tài)方法的區(qū)別的。
java類可能會有很多個對象棺榔,但是只有1個Class對象瓶堕,也就是說類的不同實例之間共享該類的Class對象。Class對象其實也僅僅是1個java對象症歇,只不過有點特殊而已郎笆。由于每個java對象都有1個互斥鎖,而類的靜態(tài)方法是需要Class對象忘晤。所以所謂的類鎖宛蚓,不過是Class對象的鎖而已。獲取類的Class對象有好幾種设塔,最簡單的就是[類名.class]的方式凄吏。
下面是類鎖的兩種不同形式的寫法:
public class TestSynchronized2{
//直接在靜態(tài)方法前加synchronized
public static synchornized void method1(){
System.out.println("This is a test");
}
//將TestSynchronized2.class作為鎖對象
public static void method2(){
synchornized(TestSynchronized2.calss){
System.out.println("This is a test");
}
}
}
總結(jié)
本質(zhì)上, synchronized 關(guān)鍵字會鎖定一個對象(當(dāng)修飾一個方法時闰蛔,會獲鎖定當(dāng)前類的實例對象this痕钢,當(dāng)修飾代碼塊時,會鎖定指定的任意對象序六,可以是Class對象任连,也可以是類中新new出來的對象),而鎖定的這個對象的生命周期決定了這個鎖互斥訪問的范圍例诀。