Java5併發庫之條件阻塞Condition的應用

鎖只能實現互斥,不能實現線程之間的通訊(能夠參看一個線程間通訊的例子http://020618.blog.51cto.com/6098149/1182807),而Condition能夠解決這個問題,即便CPU分給了當前的線程,可是當前的線程能夠讓出CPU,讓其它的相稱執行,當前線程暫停,當其它的線程執行完後,在通知當前的線程,這樣就實現了線程之間的通訊。java

Java5併發庫中的Condition的功能相似在傳統線程技術中的Object.wait()和Object.notify()的功能,在等待Condition時,容許發生「虛假喚醒」,這一般做爲對基礎平臺語義的讓步。對於大多數應用程序,這實際帶來的影響很小,由於Condition應該老是在一個循環中等待,並測試正被等待的狀態聲明。某個實現能夠隨意移除可能的虛假喚醒,但建議應用程序員老是這些虛假喚醒可能發生,所以,老是在下一個循環中等待。程序員

雖然Condition的功能相似在傳統線程技術中的Object.wait()和Object.notify()的功能,可是Condition能夠實現synchronizd實現不了的功能。併發

下面看一個Condition的代碼例子:實現3個線程時間的交替執行,使main線程喚sub2線程,sub2線程喚醒sub3線程,sub3線程喚醒main線程,如此循環50次。代碼以下:ide

  • 1-1 一個Condition的例子
  •    
       
       
       
    1. import java.util.concurrent.locks.Condition; 
    2. import java.util.concurrent.locks.Lock; 
    3. import java.util.concurrent.locks.ReentrantLock; 
    4.  
    5. public class ThreeConditionCommunication { 
    6.     public static void main(String[] args) { 
    7.         final Business business = new Business(); 
    8.         new Thread( 
    9.                 new Runnable() { 
    10.                     @Override 
    11.                     public void run() { 
    12.                         for(int i=1;i<=50;i++){ 
    13.                             business.sub2(i); 
    14.                         } 
    15.                     } 
    16.                 } 
    17.         ).start(); 
    18.          
    19.         new Thread( 
    20.                 new Runnable() { 
    21.                     @Override 
    22.                     public void run() { 
    23.                         for(int i=1;i<=50;i++){ 
    24.                             business.sub3(i); 
    25.                         } 
    26.                     } 
    27.                 } 
    28.         ).start();       
    29.          
    30.         for(int i=1;i<=50;i++){ 
    31.             business.main(i); 
    32.         } 
    33.          
    34.     } 
    35. /* 
    36.  main線程喚sub2線程,sub2線程喚醒sub3線程,sub3線程喚醒main線程,須要三個條件 
    37.  */ 
    38.     static class Business { 
    39.             Lock lock = new ReentrantLock(); 
    40.             Condition condition1 = lock.newCondition(); 
    41.             Condition condition2 = lock.newCondition(); 
    42.             Condition condition3 = lock.newCondition(); 
    43.           private int shouldSub = 1;//剛開始main線程先執行 
    44.           public  void sub2(int i){//子線程2 
    45.               lock.lock(); 
    46.               try
    47.                   while(shouldSub != 2){ 
    48.                       try { 
    49.                         condition2.await();//sub2線程在等待主線程 
    50.                     } catch (Exception e) { 
    51.                         e.printStackTrace(); 
    52.                     } 
    53.                   } 
    54.                     for(int j=1;j<=10;j++){ 
    55.                         System.out.println("sub2 thread sequence of " + j + ",loop of " + i); 
    56.                     } 
    57.                   shouldSub = 3;//執行完以後通知sub3線程 
    58.                   condition3.signal();//通知sub3線程開始執行 
    59.               }finally
    60.                   lock.unlock(); 
    61.               } 
    62.           } 
    63.  
    64.           public  void sub3(int i){//子線程3 
    65.               lock.lock(); 
    66.               try
    67.                   while(shouldSub != 3){//不應sub3線程執行 
    68.                       try { 
    69.                         condition3.await();//sub3線程執行 
    70.                     } catch (Exception e) { 
    71.                         e.printStackTrace(); 
    72.                     } 
    73.                   } 
    74.                     for(int j=1;j<=20;j++){ 
    75.                         System.out.println("sub3 thread sequence of " + j + ",loop of " + i); 
    76.                     } 
    77.                   shouldSub = 1;//執行完以後通知main線程 
    78.                   condition1.signal();//main線程開始執行 
    79.               }finally
    80.                   lock.unlock(); 
    81.               } 
    82.           }        
    83.            
    84.           public  void main(int i){//主線程 
    85.               lock.lock(); 
    86.               try
    87.                  while(shouldSub != 1){//不應main線程執行 
    88.                         try { 
    89.                             condition1.await();//main線程等待 
    90.                         } catch (Exception e) { 
    91.                             e.printStackTrace(); 
    92.                         } 
    93.                     } 
    94.                     for(int j=1;j<=100;j++){ 
    95.                         System.out.println("main thread sequence of " + j + ",loop of " + i); 
    96.                     } 
    97.                     shouldSub = 2;//main線程通知sub2線程執行 
    98.                     condition2.signal();//sub2線程按開始執行 
    99.           }finally
    100.               lock.unlock(); 
    101.           } 
    102.       } 
    103.     } 

程序運行的結果:oop

上面的代碼是在 一個線程通訊的例子(http://020618.blog.51cto.com/6098149/1182807)的基礎上進行改寫的,感興趣的朋友能夠看一下。測試

相關文章
相關標籤/搜索