標籤:
我們通過一個實際的例子來解釋Condition的用法:
我們要列印1到9這9個數字,由A線程先列印1,2,3,然後由B線程列印4,5,6,然後再由A線程列印7,8,9. 這道題有很多種解法,現在我們使用Condition來做這道題(使用Object的wait,notify方法的解法在這裡)。
package cn.xband.locks;import java.util.concurrent.locks.Condition;import java.util.concurrent.locks.Lock;import java.util.concurrent.locks.ReentrantLock;public class Test{static class NumberWrapper {public int value = 1;}public static void main(String[] args) {//初始化可重新進入鎖final Lock lock = new ReentrantLock();//第一個條件當螢幕上輸出到3final Condition reachThreeCondition = lock.newCondition();//第二個條件當螢幕上輸出到6final Condition reachSixCondition = lock.newCondition();//NumberWrapper只是為了封裝一個數字,一邊可以將數字對象共用,並可以設定為final//注意這裡不要用Integer, Integer 是不可變對象final NumberWrapper num = new NumberWrapper();//初始化A線程Thread threadA = new Thread(new Runnable() {@Overridepublic void run() {//需要先獲得鎖lock.lock();try {System.out.println("threadA start write");//A線程先輸出前3個數while (num.value <= 3) {System.out.println(num.value);num.value++;}//輸出到3時要signal,告訴B線程可以開始了reachThreeCondition.signal();} finally {lock.unlock();}lock.lock();try {//等待輸出6的條件reachSixCondition.await();System.out.println("threadA start write");//輸出剩餘數字while (num.value <= 9) {System.out.println(num.value);num.value++;}} catch (InterruptedException e) {e.printStackTrace();} finally {lock.unlock();}}});Thread threadB = new Thread(new Runnable() {@Overridepublic void run() {try {lock.lock();while (num.value <= 3) {//等待3輸出完畢的訊號reachThreeCondition.await();}} catch (InterruptedException e) {e.printStackTrace();} finally {lock.unlock();}try {lock.lock();//已經收到訊號,開始輸出4,5,6System.out.println("threadB start write");while (num.value <= 6) {System.out.println(num.value);num.value++;}//4,5,6輸出完畢,告訴A線程6輸出完了reachSixCondition.signal();} finally {lock.unlock();}}});//啟動兩個線程threadB.start();threadA.start();}}
上述代碼中有完整的注釋,請參考注釋,理解Condition的用法。
基本思路就是首先要A線程先寫1,2,3,這時候B線程應該等待reachThredCondition訊號,而當A線程寫完3之後就通過signal告訴B線程“我寫到3了,該你了”,這時候A線程要等嗲reachSixCondition訊號,同時B線程得到通知,開始寫4,5,6,寫完4,5,6之後B線程通知A線程reachSixCondition條件成立了,這時候A線程就開始寫剩下的7,8,9了。
java並發控制:ReentrantLock Condition使用詳解