2015-02-11 137 views
2

在以下簡單示例代碼中(主要取自此nice udemy video),有兩個線程通過​​方法遞增計數實例變量。for循環中的同步方法

但是,此方法在每個線程的for循環內。

我的問題是:在這種情況下,線程是否有可能將它們的調用混合到increment?例如是這樣的:

線程1個呼叫增量對於i = 0

線程1個呼叫增量對於i = 1個

線程2呼叫增量對於j = 0

線程2個呼叫1 =

線程1調用增量爲i = 2

線程2要求對於j增量= 2

...等等

我試圖運行它多次爲大型i的和j的,但他們總是順序顯示,即

線程1個呼叫增量對於i = 0

線程1個呼叫增量對於i = 1

線程1級的呼叫中crement對於i = 2

...線索1點結束

線程2呼叫增量對於j = 0

線程2個呼叫增量對於j = 1個

...線程2個完成

根據我對​​關鍵字的理解,第一種情況應該會發生,所以我只是想知道是否由於有其他事情的機會而看到第二種情況我錯過了。

public class SyncTest { 

    private int count = 0; 

    public static void main(String[] args) { 
     SyncTest test = new SyncTest(); 
     test.doWork(); 
    } 

    public synchronized void increment(String threadName) { 
     System.out.println("thread: " + threadName); 
     count++; 
    } 

    public void doWork() { 

     Thread t1 = new Thread(new Runnable() { 
      public void run() { 
       for (int i = 0; i < 100000; i++) { 
        increment("t1"); 
       } 
      } 
     }); 

     Thread t2 = new Thread(new Runnable() { 
      public void run() { 
       for (int j = 0; j < 100000; j++) { 
        increment("t2"); 
       } 
      } 
     }); 

     t1.start(); 
     t2.start(); 

     try { 
      t1.join(); 
      t2.join(); 
     } catch(InterruptedException e) { 
      e.printStackTrace(); 
     } 

     System.out.println("Count is: " + count); 
    } 
} 
+0

這個例子是正確的,你爲什麼認爲它會出錯? 'synchronized'保護整個函數體,所以這個函數被順序地調用。 – vbezhenar 2015-02-11 21:17:07

+0

我的猜測是第一個線程在第二個線程開始之前完成。整個程序運行多久?此外,您列出的輸出與發佈的程序不匹配。 – 2015-02-11 21:22:20

+0

「然而這個方法在for循環中」該方法在for循環中被調用_called_。該方法在SyncTest類的頂級_defined_中。 – 2015-02-11 22:17:21

回答

2

基於我對synchronized關鍵字的理解,第一種情況應該發生,所以我只是想知道如果我看到由於偶然第二種情況是有別的,我失去了一些東西。

你的理解是正確的。

事實上,當我跑你確切的代碼,我得到的t1秒,然後通過t2 s,則t1又在發,然後t2 S和等等的序列的核酸。這清楚地表明線程正在搶佔對方,並且沒有我們忽略的微妙的意外同步。