隨筆 - 312, 文章 - 14, 評(píng)論 - 1393, 引用 - 0
          數(shù)據(jù)加載中……

          Java多線程初學(xué)者指南(12):使用Synchronized塊同步變量

          本文為原創(chuàng),如需轉(zhuǎn)載,請(qǐng)注明作者和出處,謝謝!

          上一篇:Java多線程初學(xué)者指南(11):使用Synchronized塊同步方法

          我們可以通過(guò)synchronized塊來(lái)同步特定的靜態(tài)或非靜態(tài)方法。要想實(shí)現(xiàn)這種需求必須為這些特性的方法定義一個(gè)類(lèi)變量,然后將這些方法的代碼用synchronized塊括起來(lái),并將這個(gè)類(lèi)變量作為參數(shù)傳入synchronized塊。下面的代碼演示了如何同步特定的類(lèi)方法:

            001  package mythread;
            
          002  
            
          003  public class SyncThread extends Thread
            
          004  {
            
          005      private static String sync = "";
            
          006      private String methodType = "";
            
          007  
            
          008      private static void method(String s)
            
          009      {
            
          010          synchronized (sync)
            
          011          {
            
          012              sync = s;
            
          013              System.out.println(s);
            
          014              while (true);
            
          015          }
            
          016      }
            
          017      public void method1()
            
          018      {
            
          019          method("method1");
            
          020      }
            
          021      public static void staticMethod1()
            
          022      {
            
          023          method("staticMethod1");
            
          024      }
            
          025      public void run()
            
          026      {
            
          027          if (methodType.equals("static"))
            
          028              staticMethod1();
            
          029          else if (methodType.equals("nonstatic"))
            
          030              method1();
            
          031      }
            
          032      public SyncThread(String methodType)
            
          033      {
            
          034          this.methodType = methodType;
            
          035      }
            
          036      public static void main(String[] args) throws Exception
            
          037      {
            
          038          SyncThread sample1 = new SyncThread("nonstatic");
            
          039          SyncThread sample2 = new SyncThread("static");
            
          040          sample1.start();
            
          041          sample2.start();
            
          042      }
            
          043  }

          運(yùn)行結(jié)果如下:

          method1
          staticMethod1

          看到上面的運(yùn)行結(jié)果很多讀者可能感到驚奇。在上面的代碼中method1staticMethod1方法使用了靜態(tài)字符串變量sync進(jìn)行同步。這兩個(gè)方法只能有一個(gè)同時(shí)執(zhí)行,而這兩個(gè)方法都會(huì)執(zhí)行014行的無(wú)限循環(huán)語(yǔ)句。因此,輸出結(jié)果只能是method1staticMethod1其中之一。但這個(gè)程序?qū)⑦@兩個(gè)字符串都輸出了。

          出現(xiàn)這種結(jié)果的愿意很簡(jiǎn)單,我們看一下012行就知道了。原來(lái)在這一行將sync的值改變了。在這里要說(shuō)一下Java中的String類(lèi)型。String類(lèi)型和Java中其他的復(fù)雜類(lèi)型不同。在使用String型變量時(shí),只要給這個(gè)變量賦一次值,Java就會(huì)創(chuàng)建個(gè)新的String類(lèi)型的實(shí)例。如下面的代碼所示:

          String s = "hello";
          System.out.println(s.hashCode());
          = "world";
          System.out.println(s.hashCode()); 

          在上面的代碼中。第一個(gè)s和再次賦值后的shashCode的值是不一樣的。由于創(chuàng)建String類(lèi)的實(shí)例并不需要使用new,因此,在同步String類(lèi)型的變量時(shí)要注意不要給這個(gè)變量賦值,否則會(huì)使變量無(wú)法同步。

          由于在012行已經(jīng)為sync創(chuàng)建了一個(gè)新的實(shí)例,假設(shè)method1先執(zhí)行,當(dāng)method1方法執(zhí)行了013行的代碼后,sync的值就已經(jīng)不是最初那個(gè)值了,而method1方法鎖定的仍然是sync變量最初的那個(gè)值。而在這時(shí),staticMethod1正好執(zhí)行到synchronized(sync),在staticMethod1方法中要鎖定的這個(gè)syncmethod1方法鎖定的sync已經(jīng)不是一個(gè)了,因此,這兩個(gè)方法的同步性已經(jīng)被破壞了。

          解決以上問(wèn)題的方法當(dāng)然是將012行去掉。在本例中加上這行,只是為了說(shuō)明使用類(lèi)變量來(lái)同步方法時(shí)如果在synchronized塊中將同步變量的值改變,就會(huì)破壞方法之間的同步。為了徹底避免這種情況發(fā)生,在定義同步變量時(shí)可以使用final關(guān)鍵字。如將上面的程序中的005行可改成如下形式:

          private final static String sync = "";

              使用final關(guān)鍵字后,sync只能在定義時(shí)為其賦值,并且以后不能再修改。如果在程序的其他地方給sync賦了值,程序就無(wú)法編譯通過(guò)。在Eclipse等開(kāi)發(fā)工具中,會(huì)直接在錯(cuò)誤的地方給出提示。

              我們可以從兩個(gè)角度來(lái)理解synchronized塊。如果從類(lèi)方法的角度來(lái)理解,可以通過(guò)類(lèi)變量來(lái)同步相應(yīng)的方法。如果從類(lèi)變量的角度來(lái)理解,可以使用synchronized塊來(lái)保證某個(gè)類(lèi)變量同時(shí)只能被一個(gè)方法訪問(wèn)。不管從哪個(gè)角度來(lái)理解,它們的實(shí)質(zhì)都是一樣的,就是利用類(lèi)變量來(lái)獲得同步鎖,通過(guò)同步鎖的互斥性來(lái)實(shí)現(xiàn)同步。

          注意:在使用synchronized塊時(shí)應(yīng)注意,synchronized塊只能使用對(duì)象作為它的參數(shù)。如果是簡(jiǎn)單類(lèi)型的變量(int、char、boolean),不能使用synchronized來(lái)同步。





          Android開(kāi)發(fā)完全講義(第2版)(本書(shū)版權(quán)已輸出到臺(tái)灣)

          http://product.dangdang.com/product.aspx?product_id=22741502



          Android高薪之路:Android程序員面試寶典 http://book.360buy.com/10970314.html


          新浪微博:http://t.sina.com.cn/androidguy   昵稱:李寧_Lining

          posted on 2009-03-22 21:26 銀河使者 閱讀(7339) 評(píng)論(4)  編輯  收藏 所屬分類(lèi): java 、 原創(chuàng) 、多線程

          評(píng)論

          # re: Java多線程初學(xué)者指南(12):使用Synchronized塊同步變量  回復(fù)  更多評(píng)論   

          不知是我理解錯(cuò)誤,我的感覺(jué)是 解決以上問(wèn)題的方法當(dāng)然是將013行去掉 改成12行吧,不管那就sysout的作用吧,請(qǐng)指正
          2009-07-23 23:02 | 李峰

          # re: Java多線程初學(xué)者指南(12):使用Synchronized塊同步變量  回復(fù)  更多評(píng)論   

          Synchronized鎖定的只能是對(duì)象:this當(dāng)前對(duì)象或某類(lèi)對(duì)象obj,當(dāng)有一個(gè)線程進(jìn)入該方法時(shí),就會(huì)鎖定this或obj,這樣此時(shí)其它線程是不能進(jìn)入任何聲明對(duì)this或obj進(jìn)行鎖定的方法的,因?yàn)镾ynchronized不允許對(duì)同一對(duì)象加多把鎖的。從這里可以看出一個(gè)方法聲明Synchronized某個(gè)類(lèi)的對(duì)象引用obj時(shí),并不能保證同時(shí)只能有一個(gè)線程運(yùn)行在這個(gè)方法內(nèi),只要后來(lái)的線程改變了前一個(gè)正在運(yùn)行時(shí)obj所指向的對(duì)象,該線程也可以進(jìn)入該方法中。所有只有保證方法Synchronized的對(duì)象不發(fā)生改變才能起到同步的效果
          2009-10-23 11:16 | 張曉飛

          # re: Java多線程初學(xué)者指南(12):使用Synchronized塊同步變量  回復(fù)  更多評(píng)論   

          13行是否應(yīng)為:System.out.println(sync);
          2014-03-14 17:45 | 燕燕

          # re: Java多線程初學(xué)者指南(12):使用Synchronized塊同步變量  回復(fù)  更多評(píng)論   

          我想錯(cuò)了
          2014-03-14 18:44 | 燕燕
          主站蜘蛛池模板: 四子王旗| 武隆县| 邛崃市| 涞水县| 屏边| 岳西县| 巧家县| 咸丰县| 北辰区| 缙云县| 房产| 右玉县| 全椒县| 绥化市| 永修县| 青阳县| 蓬莱市| 九江市| 上栗县| 太谷县| 澄江县| 新丰县| 三明市| 汝阳县| 舞阳县| 陆河县| 沂水县| 逊克县| 澜沧| 安仁县| 图们市| 贡嘎县| 广德县| 大渡口区| 齐齐哈尔市| 江华| 芦溪县| 黄梅县| 万盛区| 汕尾市| 雷波县|