Java線程之線程的同步與鎖

一、同步問題提出

線程的同步是爲了防止多個線程訪問一個數據對象時,對數據造成的破壞。

二、同步和鎖定

Java中每個對象都有一個內置鎖。

當線程運行到非靜態的synchronized同步方法上時,自動獲得與正在執行代碼的當前實例(this實例)有關的鎖。獲得一個對象的鎖也稱爲獲取鎖,鎖定對象、在對象上鎖或在對象上同步。

當程序運行到synchronized同步方法貨代碼塊時才該對象鎖才起作用。

一個對象只有一個鎖。所以如果一個線程獲得該鎖,就沒有其他線程可以獲得鎖,知道第一個線程釋放(或返回)鎖。這也意味着任何其他線程不能進入該對象上synchronized方法或者代碼塊,知道該鎖被釋放。

釋放鎖是指鎖線程退出了synchronized同步方法或代碼塊。

有關鎖和同步:

a、只能同步方法、不能同步變量和類;

b、每個對象只有一個鎖;當提到同步時,應該清楚在什麼上同步?也就是說,在哪個對象上同步?

c、不必同步類中所有方法,類可以同時擁有同步和非同步方法。

d、如果兩個線程要執行一個類中的synchronized方法,並且兩個線程使用相同的實例來調用方法,那麼一次只能有一個線程能夠執行方法,另一個需要等待,直到鎖被釋放。也就是說:如果一個線程在對象上獲得一個鎖,就沒有任何其他線程可以進入(該對象的)類中的任何一個同步方法。

e、如果線程擁有同步和非同步方法,則非同步方法可以被多個線程自由訪問而不受鎖的限制。

f、線程睡眠時、它所持的任何鎖都不會釋放。

g、線程可以獲得多個鎖。比如,在一個對象的同步方法裏面調用另外一個對象的同步方法,則獲取了兩個對象的同步鎖。

h、同步損害併發性,應該盡力可能縮小同步範圍。同步不但可以同步整個方法,還可以同步方法中一部分代碼塊。

i、在使用同步代碼塊時候,應該指定在哪裏個對象上同步,也就是說要獲取那個對象的鎖。


三、靜態方法同步

要同步靜態方法,需要一個用於整個類對象的鎖,這個對象是就是這個類(XXX.class)

        public static synchronized void setName(String name) {
		XXX.name = name;
	}

	public static void setName_(String name) {
		synchronized (XXX.class) {
			XXX.name = name;
		}
	}
 四、如果線程不能不能獲得鎖會怎麼樣

如果線程試圖進入同步方法,而其鎖已經被佔用,則線程在該對象上被阻塞。實質上,線程進入該對象的的一種池中,必須在哪裏等待,直到其鎖被釋放,該線程再次變爲可運行或運行爲止。

當考慮阻塞時,一定要注意哪個對象正被用於鎖定:

1、調用同一個對象中非靜態同步方法的線程將彼此阻塞。如果是不同對象,則每個線程有自己的對象的鎖,線程間彼此互不干預。

2、調用同一個類中的靜態同步方法的線程將彼此阻塞,它們都是鎖定在相同的Class對象上。

3、靜態同步方法和非靜態同步方法將永遠不會彼此阻塞,因爲靜態方法鎖定在Class對象上,非靜態方法鎖定在該類的對象上。

4、對於同步代碼塊,要看清楚什麼對象已經用於鎖定(synchronized後面括號的內容)。在同一個對象上進行同步的線程將彼此阻塞,在不同對象上鎖定的線程將永遠不會彼此阻塞。

五、何時需要同步

在多個線程同時訪問互斥(可交換)數據時,應該同步以保護數據,確保兩個線程不會同時修改更改它。

對於非靜態字段中可更改的數據,通常使用非靜態方法訪問。

對於靜態字段中可更改的數據,通常使用靜態方法訪問。

如果需要在非靜態方法中使用靜態字段,或者在靜態字段中調用非靜態方法,問題將變得非常複雜。已經超出SJCP考試範圍了。

六、線程安全類

當一個類已經很好的同步以保護它的數據時,這個類就稱爲“線程安全的”。

即使是線程安全類,也應該特別小心,因爲操作的線程之間仍然不一定安全。

舉個形象的例子,比如一個集合是線程安全的,有兩個線程在操作同一個集合對象,當第一個線程查詢集合非空後,刪除集合中所有元素的時候。第二個線程也來執行與第一個線程相同的操作,也許在第一個線程查詢後,第二個線程也查詢出集合非空,但是當第一個執行清除後,第二個再執行刪除顯然是不對的,因爲此時集合已經爲空了。

七、線程死鎖

死鎖對Java程序來說,是很複雜的,也很難發現問題。當兩個線程被阻塞,每個線程在等待另一個線程時就發生死鎖。

八、線程同步小結

1、線程同步的目的是爲了保護多個線程反問一個資源時對資源的破壞。

2、線程同步方法是通過鎖來實現,每個對象都有切僅有一個鎖,這個鎖與一個特定的對象關聯,線程一旦獲取了對象鎖,其他訪問該對象的線程就無法再訪問該對象的其他同步方法。

3、對於靜態同步方法,鎖是針對這個類的,鎖對象是該類的Class對象。靜態和非靜態方法的鎖互不干預。一個線程獲得鎖,當在一個同步方法中訪問另外對象上的同步方法時,會獲取這兩個對象鎖。

4、對於同步,要時刻清醒在哪個對象上同步,這是關鍵。

5、編寫線程安全的類,需要時刻注意對多個線程競爭訪問資源的邏輯和安全做出正確的判斷,對“原子”操作做出分析,並保證原子操作期間別的線程無法訪問競爭資源。

6、當多個線程等待一個對象鎖時,沒有獲取到鎖的線程將發生阻塞。

7、死鎖是線程間相互等待鎖鎖造成的,在實際中發生的概率非常的小。真讓你寫個死鎖程序,不一定好使,呵呵。但是,一旦程序發生死鎖,程序將死掉。

發佈了68 篇原創文章 · 獲贊 1 · 訪問量 5928
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章