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