java - 一個類的對象鎖只有一個,類鎖呢?
問題描述
一個類的對象鎖只有一個,如果有幾個非靜態(tài)函數(shù)都是synchronized,在某一時刻只有一個線程能調(diào)用其中一個函數(shù)
假如一個類有幾個靜態(tài)函數(shù)是synchronized,在某一時刻只有一個線程能調(diào)用其中一個靜態(tài)函數(shù)嗎?也就是類鎖也只有一個嗎?
問題解答
回答1:前面一種鎖的是實例對象,鎖定了當(dāng)前的那個對象,如果有多個實例對象,這些synchronized方法之間不是同步的。第二種鎖的是類對象,類對象就一個,所以是同步的。
回答2:public class Foo { synchronized void test1() {//to implements } void test2() {synchronized(this) { //to implements} } synchronized static void test3() {//to implements } static void test4() {synchronized(Foo.class) { //to implements} }}
如上代碼,test1方法相當(dāng)于test2,當(dāng)this是同一個對象時,會發(fā)生阻塞。當(dāng)然,不同對象沒有關(guān)系,因為this不一樣。稱為對象級鎖。test3相當(dāng)于test4,這里是用class對象作為鎖,因為一般情況下一個類的類實例只有一個,那么每次進入這個方法都會鎖。稱為類級鎖。
回答3:非靜態(tài)同步方法(A)用的鎖就是當(dāng)前實例對象本身,一個實例的A獲取鎖之后,該實例的其他A必須等待鎖的釋放,多個實例用的都是不同的鎖;
靜態(tài)同步方法(B)用的鎖是類對象本身,一旦一個B獲取鎖之后其他的B都必須等待釋放鎖,不管是一個實例還是多個實例;
另外 A和B之間用不同的鎖,所以不會有競爭關(guān)系;
回答4:Class類創(chuàng)建一個對象就是代表一個普通類,這時“類鎖”就是這個實例對象上的鎖
回答5:你說的是“互斥鎖”的概念,針對synchronized修飾方法有兩種情況:
【非靜態(tài)方法】
當(dāng)一個方法被synchronized修飾后,鎖對象為當(dāng)前方法所屬對象,即方法中的this。
【靜態(tài)方法】
當(dāng)一個靜態(tài)方法被synchronized修飾后,該靜態(tài)方法上鎖的對象為當(dāng)前類對象(Class類的實例)。每個類都有唯一的一個類對象。獲取類對象的方式:類名.class。
而對于互斥的場景,需要理解兩點說明:
1、靜態(tài)方法與非靜態(tài)方法同時聲明了synchronized,他們之間是非互斥關(guān)系的。原因在于,靜態(tài)方法鎖的是類對象而非靜態(tài)方法鎖的是當(dāng)前方法所屬對象。
2、當(dāng)Synchronized修飾的是兩段不同的代碼,但是鎖對象相同時,兩個線程分別調(diào)用者兩段代碼時就是互斥的
所以你說的“一時刻只有一個線程能調(diào)用其中一個函數(shù)”(即互斥),判斷條件就是鎖對象是否相同,與方法類型無關(guān)。
回答6:對象方法的synchronized修飾,鎖為對象自身,也就是this;靜態(tài)方法的synchronized修飾,鎖為Class對象自身,也就是由類加載器創(chuàng)建的類對象;

網(wǎng)公網(wǎng)安備