簡單記錄下java中synchronized關鍵字的使用方法。
在介紹之前需要明確下java中的每一個類別的物件實例都有且只有一個鎖(lock)和之相關聯,synchronized關鍵字只作用於該鎖,即可以認為synchronized只對java類別的物件實例起作用。
synchronized修飾函數
複製代碼代碼如下:
public synchronized aMethod(){
}
這就是最常用的情景,那麼這個同步方法的用途是啥,為了方便就記作aMethod方法。
1.synchronized鎖定的是呼叫這個同步方法的物件實例,舉個例子,同一個實例P1在不同執行緒中都呼叫aMethod時會產生同步;
2.需要注意的是這個物件所屬的類別的另一物件P2卻能夠任意呼叫這個aMethod,因為不同的物件實例的synchronized方法是不相幹擾的。也就是說,其它執行緒照樣可以同時存取相同類別的另一個物件實例中的aMethod方法;
3.如果一個物件有多個synchronized方法,例如aMethod、bMethod、cMethod,現在只要一個物件存取了其中的一個synchronized方法,其它執行緒不能同時存取這個物件中任何一個synchronized方法。
上述程式碼其實等價於下面:
複製代碼代碼如下:
public void aMethod() {
synchronized (this) {
}
}
這裡的this就是指的該實例物件的引用,如P1。可見同步方法實質是將synchronized作用於object reference。那個拿到了P1物件鎖的線程,才能夠呼叫P1的同步方法,而對P2而言,P1這個鎖和他毫不相干,程式也可能在這種情況下擺脫同步機制的控制,造成資料混亂。由此我們引出了下面的同步塊。
synchronized修飾程式碼區塊
複製代碼代碼如下:
public void dMethod(SomeObject so) {
synchronized(so) {
}
}
這裡synchronized取得鎖就是so這個物件的鎖,因而誰拿到這個鎖誰就能夠運行他所控制的那段程式碼。當有一個明確的物件作為鎖時,就能夠這樣寫程式,但當沒有明確的物件作為鎖,只是想讓一段程式碼同步時,能夠創建一個特別的instance變數(他得是個物件)來充當鎖:
複製代碼代碼如下:
class Foo implements Runnable {
private byte[] lock = new byte[0];
Public void method() {
synchronized(lock) {
}
}
}
零長度的byte數組物件創建起來將比任何物件都更經濟高效。
synchronized修飾靜態方法
前面提到了synchronized關鍵字只對不同執行緒中的P1實例有效,那如何可以同時對P1和P2不同實例有效呢,答案就是使用synchronized修飾靜態方法,類別的靜態方法可以說是這個類別自有的,不依賴類別的實例,所以我們只要對類別的靜態方法使用synchronized關鍵字來修飾就可以達到不同實例間的同步了。