使用Synchronized块同步方法
synchronized關(guān)鍵字有兩種用法。第一種就是在《使用Synchronized關(guān)鍵字同步類方法》一文中所介紹的直接用在方法的定義中。另外一種就是synchronized塊。我們不僅可以通過synchronized塊來同步一個對象變量。也可以使用synchronized塊來同步類中的靜態(tài)方法和非靜態(tài)方法。
synchronized塊的語法如下:
public void method() { … … synchronized(表達式) { … … } }一、非靜態(tài)類方法的同步
從《使用Synchronized關(guān)鍵字同步類方法》一文中我們知道使用synchronized關(guān)鍵字來定義方法就會鎖定類中所有使用synchronzied關(guān)鍵字定義的靜態(tài)方法或非靜態(tài)方法,但這并不好理解。而如果使用synchronized塊來達到同樣的效果,就不難理解為什么會產(chǎn)生這種效果了。如果想使用synchronized塊來鎖定類中所有的同步非靜態(tài)方法,需要使用this做為synchronized塊的參數(shù)傳入synchronized塊國,代碼如下:
通過synchronized塊同步非靜態(tài)方法
public class SyncBlock { public void method1() { synchronized(this) // 相當(dāng)于對method1方法使用synchronized關(guān)鍵字 { … … } } public void method2() { synchronized(this) // 相當(dāng)于對method2方法使用synchronized關(guān)鍵字 { … … } } public synchronized void method3() { … … } }在上面的代碼中的method1和method2方法中使用了synchronized塊。而第017行的method3方法仍然使用synchronized關(guān)鍵字來定義方法。在使用同一個SyncBlock類實例時,這三個方法只要有一個正在執(zhí)行,其他兩個方法就會因未獲得同步鎖而被阻塞。在使用synchronized塊時要想達到和synchronized關(guān)鍵字同樣的效果,必須將所有的代碼都寫在synchronized塊中,否則,將無法使當(dāng)前方法中的所有代碼和其他的方法同步。
除了使用this做為synchronized塊的參數(shù)外,還可以使用SyncBlock.this作為synchronized塊的參數(shù)來達到同樣的效果。
在內(nèi)部類(InnerClass)的方法中使用synchronized塊來時,this只表示內(nèi)類,和外類(OuterClass)沒有關(guān)系。但內(nèi)類的非靜態(tài)方法可以和外類的非靜態(tài)方法同步。如在內(nèi)類InnerClass中加一個method4方法,并使method4方法和SyncBlock的三個方法同步,代碼如下:
使內(nèi)類的非靜態(tài)方法和外類的非靜態(tài)方法同步
public class SyncBlock { … … class InnerClass { public void method4() { synchronized(SyncBlock.this) { … … } } } … … }在上面SyncBlock類的新版本中,InnerClass類的method4方法和SyncBlock類的其他三個方法同步,因此,method1、method2、method3和method4四個方法在同一時間只能有一個方法執(zhí)行。
Synchronized塊不管是正常執(zhí)行完,還是因為程序出錯而異常退出synchronized塊,當(dāng)前的synchronized塊所持有的同步鎖都會自動釋放。
因此,在使用synchronized塊時不必擔(dān)心同步鎖的釋放問題。
二、靜態(tài)類方法的同步
由于在調(diào)用靜態(tài)方法時,對象實例不一定被創(chuàng)建。因此,就不能使用this來同步靜態(tài)方法,而必須使用Class對象來同步靜態(tài)方法。代碼如下:
通過synchronized塊同步靜態(tài)方法
public class StaticSyncBlock { public static void method1() { synchronized(StaticSyncBlock.class) { … … } } public static synchronized void method2() { … … } }在同步靜態(tài)方法時可以使用類的靜態(tài)字段class來得到Class對象。在上例中method1和method2方法同時只能有一個方法執(zhí)行。除了使用class字段得到Class對象外,還可以使用實例的getClass方法來得到Class對象。上例中的代碼可以修改如下:
使用getClass方法得到Class對象
public class StaticSyncBlock { public static StaticSyncBlock instance; public StaticSyncBlock() { instance = this; } public static void method1() { synchronized(instance.getClass()) { } } }在上面代碼中通過一個public的靜態(tài)instance得到一個StaticSyncBlock類的實例,并通過這個實例的getClass方法得到了Class對象(一個類的所有實例通過getClass方法得到的都是同一個Class對象,因此,調(diào)用任何一個實例的getClass方法都可以)。我們還可以通過Class對象使不同類的靜態(tài)方法同步,如Test類的靜態(tài)方法method和StaticSyncBlock類的兩個靜態(tài)方法同步,代碼如下:
Test類的method方法和StaticSyncBlock類的method1、method2方法同步
public class Test { public static void method() { synchronized(StaticSyncBlock.class) { } } }注意:在使用synchronized塊同步類方法時,非靜態(tài)方法可以使用this來同步,而靜態(tài)方法必須使用Class對象來同步。它們互不影響。當(dāng)然,也可以在非靜態(tài)方法中使用Class對象來同步靜態(tài)方法。但在靜態(tài)方法中不能使用this來同步非靜態(tài)方法。這一點在使用synchronized塊同步類方法時應(yīng)注意。
轉(zhuǎn)載于:https://www.cnblogs.com/yuyu666/p/9741434.html
總結(jié)
以上是生活随笔為你收集整理的使用Synchronized块同步方法的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 恒易贷提现隔天几点到
- 下一篇: 平安福交满20年后怎样