Java线程详解(5)-线程的同步与锁
一、同步問題提出
? ? ? ? 線程的同步是為了防止多個線程訪問一個數(shù)據(jù)對象時,對數(shù)據(jù)造成的破壞。
????????例如:兩個線程ThreadA、ThreadB都操作同一個對象Foo對象,并修改Foo對象上的數(shù)據(jù)。
public?class?Foo?{??private?int?x?=?100;??public?int?getX()?{??return?x;??}??public?int?fix(int?y)?{??x?=?x?-?y;??return?x;??}?? }???public?class?FooRunnable?implements?Runnable?{??private?Foo?foo?=new?Foo();??public?static?void?main(String[]?args)?{??FooRunnable?r?=?new?FooRunnable();??Thread?ta?=?new?Thread(r,"Thread-A");??Thread?tb?=?new?Thread(r,"Thread-B");??ta.start();??tb.start();??}??@Override??public?void?run()?{??for?(int?i?=?0;?i?<?3;?i++)?{??this.fix(30);??try?{??Thread.sleep(1);??}?catch?(InterruptedException?e)?{??e.printStackTrace();??}??System.out.println(Thread.currentThread().getName()+?"?:當前foo對象的x值=?"?+?foo.getX());??}??}??public?int?fix(int?y)?{??return?foo.fix(y);??}?? }??
????????執(zhí)行結果:
????????從結果發(fā)現(xiàn),這樣的輸出值明顯是不合理的,原因是兩個線程不加控制的訪問Foo對象并修改其數(shù)據(jù)所致。
????????如果要保持結果的合理性,只需要達到一個目的,就是將對Foo的訪問加以限制,每次只能有一個線程在訪問。這樣就能保證Foo對象中數(shù)據(jù)的合理性了。
????????在具體的Java代碼中需要完成以下兩個操作:
????????把競爭訪問的資源類Foo變量x標識為private;
????????同步修改變量的代碼,使用synchronized關鍵字同步方法或代碼。
?
二、同步和鎖定
? ? ? ? 1、鎖的原理
????????Java中每個對象都有一個內(nèi)置鎖。
????????當程序運行到非靜態(tài)的synchronized同步方法上時,自動獲得與正在執(zhí)行代碼類的當前實例(this實例)有關的鎖。獲得一個對象的鎖也稱為獲取鎖、鎖定對象、在對象上鎖定或在對象上同步。
????????當程序運行到synchronized同步方法或代碼塊時才該對象鎖才起作用。
????????一個對象只有一個鎖。所以,如果一個線程獲得該鎖,就沒有其他線程可以獲得鎖,直到第一個線程釋放(或返回)鎖。這也意味著任何其他線程都不能進入該對象上的synchronized方法或代碼塊,直到該鎖被釋放。
????????釋放鎖是指持鎖線程退出了synchronized同步方法或代碼塊。
????????關于鎖和同步,有一下幾個要點:
????????1)只能同步方法,而不能同步變量和類;
????????2)每個對象只有一個鎖;當提到同步時,應該清楚在什么上同步?也就是說,在哪個對象上同步?
????????3)不必同步類中所有的方法,類可以同時擁有同步和非同步方法。
????????4)如果兩個線程要執(zhí)行一個類中的synchronized方法,并且兩個線程使用相同的實例來調用方法,那么一次只能有一個線程能夠執(zhí)行方法,另一個需要等待,直到鎖被釋放。也就是說:如果一個線程在對象上獲得一個鎖,就沒有任何其他線程可以進入(該對象的)類中的任何一個同步方法。
????????5)如果線程擁有同步和非同步方法,則非同步方法可以被多個線程自由訪問而不受鎖的限制。
????????6)線程睡眠時,它所持的任何鎖都不會釋放。
????????7)線程可以獲得多個鎖。比如,在一個對象的同步方法里面調用另外一個對象的同步方法,則獲取了兩個對象的同步鎖。
????????8)同步損害并發(fā)性,應該盡可能縮小同步范圍。同步不但可以同步整個方法,還可以同步方法中一部分代碼塊。
????????9)在使用同步代碼塊時候,應該指定在哪個對象上同步,也就是說要獲取哪個對象的鎖。例如:
public?int?fix(int?y)?{??synchronized?(this)?{??x?=?x?-?y;??}??return?x;??}??
????????當然,同步方法也可以改寫為非同步方法,但功能完全一樣的,例如:
????????與 ?
? ? ? ? 效果是完全一樣的。
?
三、靜態(tài)方法同步
? ? ? ? 要同步靜態(tài)方法,需要一個用于整個類對象的鎖,這個對象是就是這個類(XXX.class)。
????????例如:
public?staticsynchronized?int?setName(String?name){??Xxx.name?=?name;?? }??
????????等價于
四、如果線程不能獲得鎖會怎么樣
? ? ? ? 如果線程試圖進入同步方法,而其鎖已經(jīng)被占用,則線程在該對象上被阻塞。實質上,線程進入該對象的一種池中,必須在那里等待,直到其鎖被釋放,該線程再次變?yōu)榭蛇\行或運行為止。
????????當考慮阻塞時,一定要注意哪個對象正被用于鎖定:
????????1、調用同一個對象中非靜態(tài)同步方法的線程將彼此阻塞。如果是不同對象,則每個線程有自己的對象的鎖,線程間彼此互不干預。
????????2、調用同一個類中的靜態(tài)同步方法的線程將彼此阻塞,它們都是鎖定在相同的Class對象上。
????????3、靜態(tài)同步方法和非靜態(tài)同步方法將永遠不會彼此阻塞,因為靜態(tài)方法鎖定在Class對象上,非靜態(tài)方法鎖定在該類的對象上。
????????4、對于同步代碼塊,要看清楚什么對象已經(jīng)用于鎖定(synchronized后面括號的內(nèi)容)。在同一個對象上進行同步的線程將彼此阻塞,在不同對象上鎖定的線程將永遠不會彼此阻塞。
?
五、何時需要同步
? ? ? ? 在多個線程同時訪問互斥(可交換)數(shù)據(jù)時,應該同步以保護數(shù)據(jù),確保兩個線程不會同時修改更改它。
????????對于非靜態(tài)字段中可更改的數(shù)據(jù),通常使用非靜態(tài)方法訪問。
????????對于靜態(tài)字段中可更改的數(shù)據(jù),通常使用靜態(tài)方法訪問。
????????如果需要在非靜態(tài)方法中使用靜態(tài)字段,或者在靜態(tài)字段中調用非靜態(tài)方法,問題將變得非常復雜。
?
六、線程安全類
? ? ? ? 當一個類已經(jīng)很好的同步以保護它的數(shù)據(jù)時,這個類就稱為“線程安全的”。
????????即使是線程安全類,也應該特別小心,因為操作的線程之間仍然不一定安全。
????????舉個形象的例子,比如一個集合是線程安全的,有兩個線程在操作同一個集合對象,當?shù)谝粋€線程查詢集合非空后,刪除集合中所有元素的時候。第二個線程也來執(zhí)行與第一個線程相同的操作,也許在第一個線程查詢后,第二個線程也查詢出集合非空,但是當?shù)谝粋€執(zhí)行清除后,第二個再執(zhí)行刪除顯然是不對的,因為此時集合已經(jīng)為空了。
????????舉個例子:
public?class?NameList?{??private?List?nameList?=?Collections.synchronizedList(newLinkedList());??public?void?add(String?name)?{??nameList.add(name);??}??public?String?removeFirst()?{??if?(nameList.size()>0)?{??return?(String)?nameList.remove(0);??}?else?{??return?null;??}??}???? }??public?class?TestNameList?{??public?static?void?main(String[]?args)?{??final?NameList?nl?=new?NameList();??nl.add("蘇東坡");??class?NameDropper?extends?Thread{??@Override??public?void?run()?{??String?name?=?nl.removeFirst();??System.out.println(name);??}??????????}??Thread?t1=new?NameDropper();??Thread?t2=new?NameDropper();??t1.start();??t2.start();??}?? }??
????????執(zhí)行結果:
????????雖然集合對象
是同步的,但是程序還不是線程安全的。
????????出現(xiàn)這種事件的原因是,上例中一個線程操作列表過程中無法阻止另外一個線程對列表的其他操作。
????????解決上面問題的辦法是,在操作集合對象的NameList上面做一個同步。改寫后的代碼如下:
public?class?NameList?{??private?List?nameList?=?Collections.synchronizedList(newLinkedList());??public?synchronized?void?add(String?name)?{??nameList.add(name);??}??public?synchronized?StringremoveFirst()?{??if?(nameList.size()>0)?{??return?(String)?nameList.remove(0);??}?else?{??return?null;??}??}???? }??
????????這樣,當一個線程訪問其中一個同步方法時,其他線程只有等待。
?
七、線程死鎖
? ? ? ? 死鎖對Java程序來說,是很復雜的,也很難發(fā)現(xiàn)問題。當兩個線程被阻塞,每個線程在等待另一個線程時就發(fā)生死鎖。
????????還是看一個比較直觀的死鎖例子:
public?class?Deadlock?{??private?static?class?Resource{??public?int?value;??}??private?Resource?resourceA=new?Resource();??private?Resource?resourceB=new?Resource();??public?int?read(){??synchronized?(resourceA)?{??synchronized?(resourceB)?{??return?resourceB.value+resourceA.value;??}??}??}??public?void?write(int?a,int?b){??synchronized(resourceB){??synchronized?(resourceA)?{??resourceA.value=a;??resourceB.value=b;??}??}??}?? }??
????????假設read()方法由一個線程啟動,write()方法由另外一個線程啟動。讀線程將擁有resourceA鎖,寫線程將擁有resourceB鎖,兩者都堅持等待的話就出現(xiàn)死鎖。
????????實際上,上面這個例子發(fā)生死鎖的概率很小。因為在代碼內(nèi)的某個點,CPU必須從讀線程切換到寫線程,所以,死鎖基本上不能發(fā)生。
????????但是,無論代碼中發(fā)生死鎖的概率有多小,一旦發(fā)生死鎖,程序就死掉。有一些設計方法能幫助避免死鎖,包括始終按照預定義的順序獲取鎖這一策略。已經(jīng)超出SCJP的考試范圍。
?
八、線程同步小結
? ? ? ? 1、線程同步的目的是為了保護多個線程反問一個資源時對資源的破壞。
????????2、線程同步方法是通過鎖來實現(xiàn),每個對象都有切僅有一個鎖,這個鎖與一個特定的對象關聯(lián),線程一旦獲取了對象鎖,其他訪問該對象的線程就無法再訪問該對象的其他同步方法。
????????3、對于靜態(tài)同步方法,鎖是針對這個類的,鎖對象是該類的Class對象。靜態(tài)和非靜態(tài)方法的鎖互不干預。一個線程獲得鎖,當在一個同步方法中訪問另外對象上的同步方法時,會獲取這兩個對象鎖。
????????4、對于同步,要時刻清醒在哪個對象上同步,這是關鍵。
????????5、編寫線程安全的類,需要時刻注意對多個線程競爭訪問資源的邏輯和安全做出正確的判斷,對“原子”操作做出分析,并保證原子操作期間別的線程無法訪問競爭資源。
????????6、當多個線程等待一個對象鎖時,沒有獲取到鎖的線程將發(fā)生阻塞。
????????7、死鎖是線程間相互等待鎖鎖造成的,在實際中發(fā)生的概率非常的小。真讓你寫個死鎖程序,不一定好使,呵呵。但是,一旦程序發(fā)生死鎖,程序將死掉。
?
總結
以上是生活随笔為你收集整理的Java线程详解(5)-线程的同步与锁的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Java线程详解(4)-线程状态的转换
- 下一篇: Java线程详解(6)-线程的交互