1. 程式人生 > >java多執行緒程式設計之使用Synchronized塊同步變數

java多執行緒程式設計之使用Synchronized塊同步變數

通過synchronized塊來同步特定的靜態或非靜態方法。

要想實現這種需求必須為這些特性的方法定義一個類變數,然後將這些方法的程式碼用synchronized塊括起來,並將這個類變數作為引數傳入synchronized塊
 
下面的程式碼演示瞭如何同步特定的類方法:

public class SyncThread extends Thread
{
    private static String sync = "";
    private String methodType = "";

    public SyncThread(String methodType)
    {
       this.methodType = 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 static void main(String[] args) throws Exception
    {
        SyncThread sample1 = new SyncThread("nonstatic");
        SyncThread sample2 = new SyncThread("static");
        sample1.start();
        sample2.start();
    }
}


執行結果如下:


method1
staticMethod1


看到上面的執行結果很多讀者可能感到驚奇。
在上面的程式碼中method1和staticMethod1方法使用了靜態字串變數sync進行同步。 這兩個方法只能有一個同時執行,而這兩個方法都會執行014行的無限迴圈語句。因此,輸出結果只能是method1和staticMethod1其中之 一。但這個程式將這兩個字串都輸出了。
出現這種結果的願意很簡單,我們看一下行 sync = s; 就知道了。原來在這一行將sync的值改變了。在這裡要說 一下Java中的String型別。String型別和Java中其他的複雜型別不同。在使用String型變數時,只要給這個變數賦一次值,Java就 會建立個新的String型別的例項。如下面的程式碼所示:

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

在上面的程式碼中。第一個s和再次賦值後的s的hashCode的值是不一樣的。由於建立String類的例項並不需要使用new,因此,在同步String型別的變數時要注意不要給這個變數賦值,否則會使變數無法同步。
由於在行 sync = s 已經為sync建立了一個新的例項,假設method1先執行,當method1方法執行了 sync = s 的程式碼後,sync的值就已經不是最初那 個值了,而method1方法鎖定的仍然是sync變數最初的那個值。而在這時,staticMethod1正好執行到 synchronized(sync),在staticMethod1方法中要鎖定的這個sync和method1方法鎖定的sync已經不是一個了,因 此,這兩個方法的同步性已經被破壞了。
解決以上問題的方法當然是將該行去掉。在本例中加上這行,只是為了說明使用類變數來同步方法時如果在 synchronized塊中將同步變數的值改變,就會破壞方法之間的同步。為了徹底避免這種情況發生,在定義同步變數時可以使用final關鍵字。可改成如下形式:


private final static String sync = "";


使用final關鍵字後,sync只能在定義時為其賦值,並且以後不能再修改。如果在程式的其他地方給sync賦了值,程式就無法編譯通過。在Eclipse等開發工具中,會直接在錯誤的地方給出提示。
我 們可以從兩個角度來理解synchronized塊。如果從類方法的角度來理解,可以通過類變數來同步相應的方法。如果從類變數的角度來理解,可以使用 synchronized塊來保證某個類變數同時只能被一個方法訪問。不管從哪個角度來理解,它們的實質都是一樣的,就是利用類變數來獲得同步鎖,通過同 步鎖的互斥性來實現同步。

注意:在使用synchronized塊時應注意,synchronized塊只能使用物件作為它的引數。如果是簡單型別的變數(如int、char、boolean等),不能使用synchronized來同步.


我發現synchronized 是屬於執行緒間的同步鎖, 對main主執行緒起不了任何約束,當一個執行緒對某個物件啟用了synchronized , 其它執行緒將進入等待, 及無法操作此物件, main函式卻不需要等待,可以直接進行操作