java多線程編程之使用Synchronized塊同步變量

咱們能夠經過synchronized塊來同步特定的靜態或非靜態方法。要想實現這種需求必須爲這些特性的方法定義一個類變量,而後將這些方法的代碼用synchronized塊括起來,並將這個類變量做爲參數傳入synchronized塊
 

下面的代碼演示瞭如何同步特定的類方法:工具

複製代碼 代碼以下:

package mythread;

public class SyncThread extends Thread
{
 private static String sync = "";
 private String methodType = "";開發工具

 private static void method(String s)
 {
  synchronized (sync)
  {
sync = s;
System.out.println(s);
while (true);
  }
 }
 public void method1()
 {
  method("method1");
 }
 public static void staticMethod1()
 {
  method("staticMethod1");
 }
 public void run()
 {
  if (methodType.equals("static"))
staticMethod1();
  else if (methodType.equals("nonstatic"))
method1();
 }
 public SyncThread(String methodType)
 {
  this.methodType = methodType;
 }
 public static void main(String[] args) throws Exception
 {
  SyncThread sample1 = new SyncThread("nonstatic");
  SyncThread sample2 = new SyncThread("static");
  sample1.start();
  sample2.start();
 }
}this

運行結果以下:spa

複製代碼 代碼以下:

method1
staticMethod1

看到上面的運行結果不少讀者可能感到驚奇。在上面的代碼中method1和staticMethod1方法使用了靜態字符串變量sync進行同步。 這兩個方法只能有一個同時執行,而這兩個方法都會執行014行的無限循環語句。所以,輸出結果只能是method1和staticMethod1其中之 一。但這個程序將這兩個字符串都輸出了。
出現這種結果的願意很簡單,咱們看一下012行就知道了。原來在這一行將sync的值改變了。在這裏要說 一下Java中的String類型。String類型和Java中其餘的複雜類型不一樣。在使用String型變量時,只要給這個變量賦一次值,Java就 會建立個新的String類型的實例。以下面的代碼所示:code

複製代碼 代碼以下:

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

在上面的代碼中。第一個s和再次賦值後的s的hashCode的值是不同的。因爲建立String類的實例並不須要使用new,所以,在同步String類型的變量時要注意不要給這個變量賦值,不然會使變量沒法同步。
由 於在012行已經爲sync建立了一個新的實例,假設method1先執行,當method1方法執行了013行的代碼後,sync的值就已經不是最初那 個值了,而method1方法鎖定的仍然是sync變量最初的那個值。而在這時,staticMethod1正好執行到 synchronized(sync),在staticMethod1方法中要鎖定的這個sync和method1方法鎖定的sync已經不是一個了,因 此,這兩個方法的同步性已經被破壞了。
解決以上問題的方法固然是將012行去掉。在本例中加上這行,只是爲了說明使用類變量來同步方法時若是在 synchronized塊中將同步變量的值改變,就會破壞方法之間的同步。爲了完全避免這種狀況發生,在定義同步變量時可使用final關鍵字。如將 上面的程序中的005行可改爲以下形式:對象

複製代碼 代碼以下:

private final static String sync = "";

使用final關鍵字後,sync只能在定義時爲其賦值,而且之後不能再修改。若是在程序的其餘地方給sync賦了值,程序就沒法編譯經過。在Eclipse等開發工具中,會直接在錯誤的地方給出提示。
我 們能夠從兩個角度來理解synchronized塊。若是從類方法的角度來理解,能夠經過類變量來同步相應的方法。若是從類變量的角度來理解,可使用 synchronized塊來保證某個類變量同時只能被一個方法訪問。無論從哪一個角度來理解,它們的實質都是同樣的,就是利用類變量來得到同步鎖,經過同 步鎖的互斥性來實現同步。
注意:在使用synchronized塊時應注意,synchronized塊只能使用對象做爲它的參數。若是是簡單類型的變量(如int、char、boolean等),不能使用synchronized來同步。ip

相關文章
相關標籤/搜索