1. 程式人生 > >JDK原始碼分析--多執行緒同步工具CountDownLatch類

JDK原始碼分析--多執行緒同步工具CountDownLatch類

CountDownLatch類運用了java開發模式中的策略模式。對執行緒作用的是CountDownLatch類中的內部類Sync。

Sync類繼承了AbstractQueuedSynchronizer類,AbstractQueuedSynchronizer類是jdk多執行緒同步功能的重要類。CountDownLatch類有兩個很重要的方法:await和countDown。這兩個方法內分別呼叫的是AbstractQueuedSynchronizer的方法acquireSharedInterruptibly(int arg)和releaseShared(int arg)

CountDownLatch

類的邏輯是:使用CountDownLatch類,每個執行緒都公用一個初始化count的CountDownLatch物件常量,並在執行緒執行完成的時候呼叫countDown(),並在主執行緒使用await()進入阻塞狀態,直到所有的任務完成,當count 減到0的時候開始執行主執行緒程式碼。

應用場景是:1.一個執行緒等待多個其它執行緒執行完成。多個執行緒等待其他多個執行緒執行完成。

這種多執行緒同步工具適用於在軟體系統中,在開始某些任務前必須初始化一些系統資料,任務執行緒就必須等待系統資料初始化資料的完成,大型軟體業務系統或計算系統就需要者同步工具。

CountDownLatch中重要 的兩個方法:await()和countDown():

await():這個方法呼叫的是 sync.acquireSharedInterruptibly(1);

  public final void acquireSharedInterruptibly(int arg)
            throws InterruptedException {
        //檢驗當前執行緒是否設定了阻斷標識位為true
        if (Thread.interrupted())
            throw new InterruptedException();
        //嘗試獲取 分享模式下的鎖
        if (tryAcquireShared(arg) < 0)
            doAcquireSharedInterruptibly(arg);
    }
tryAcquireShared(int acquires)方法試圖獲取分享模式下的鎖,這個方法由CountDownLatch類中的內部類Sync類實現,方法的原始碼如下:
        protected int tryAcquireShared(int acquires) {
            return (getState() == 0) ? 1 : -1;
        }
tryAcquireShared方法的意義是:同步鎖的狀態欄位state為0,就返回-1,不為0就返回1.也就是說state的值為0,共享鎖就能被執行緒獲取,主執行緒能繼續執行,如果是大於0,則表示主執行緒不能執行,併發生阻塞,進入鎖的獲取阻塞執行緒佇列中去,可CountDownLatch類建立的時候會傳進去一個值並賦給state屬性。

共享鎖獲取成功後會執行doAcquireSharedInterruptibly(int arg)函式,該方法是將獲取鎖的等待執行緒都放入到阻塞佇列中去,原始碼如下:

    private void doAcquireSharedInterruptibly(int arg)
        throws InterruptedException {
    	//同步等待佇列中的頭節點設定為空值的Node,表示同步鎖處於共享模式
        final Node node = addWaiter(Node.SHARED);
        boolean failed = true;
        try {
            for (;;) {
                final Node p = node.predecessor();
                //如果當前執行緒為頭結點第一個後續節點,就在此檢測是否能獲取共享鎖。如果能獲取就啟用下一個節點,讓其獲得共享鎖,如果不能獲取共享鎖,就掛起。
                if (p == head) {
                    int r = tryAcquireShared(arg);//檢測是否能獲取共享鎖
                    if (r >= 0) { //這裡獲取鎖成功後,喚醒下一個AQS佇列中的下一個執行緒,但當前的後繼節點為當前執行緒,所以下面會將後繼節點設定為head。
                        setHeadAndPropagate(node, r);//啟用下一個節點,本執行緒繼續執行。
                        p.next = null; // help GC
                        failed = false;
                        return;
                    }
                }
                //如果獲取共享鎖失敗,就掛起執行緒。
                if (shouldParkAfterFailedAcquire(p, node) &&
                    parkAndCheckInterrupt())
                    throw new InterruptedException();
                //在上面if中,執行緒被掛起阻塞了,當被喚醒後會繼續執行迴圈
            }
        } finally {
            if (failed)
                cancelAcquire(node);
        }
    }
如果執行緒獲取了共享鎖,就呼叫setHeadAndPropagate()方法將下一個節點也啟用(獲取共享鎖),這個啟用的方法原始碼如下:
    private void setHeadAndPropagate(Node node, int propagate) {
        Node h = head; // Record old head for check below
        setHead(node);
        /*
         * Try to signal next queued node if:
         *   Propagation was indicated by caller,
         *     or was recorded (as h.waitStatus) by a previous operation
         *     (note: this uses sign-check of waitStatus because
         *      PROPAGATE status may transition to SIGNAL.)
         * and
         *   The next node is waiting in shared mode,
         *     or we don't know, because it appears null
         *
         * The conservatism in both of these checks may cause
         * unnecessary wake-ups, but only when there are multiple
         * racing acquires/releases, so most need signals now or soon
         * anyway.
         */
        if (propagate > 0 || h == null || h.waitStatus < 0) {
            Node s = node.next;
            if (s == null || s.isShared())
                doReleaseShared();
        }
    }
首先使用setHead方法將頭結點的後繼節點設定為頭節點,下面if語句中propagate>0表示AQS佇列上的所有執行緒都能獲取共享鎖,然後如果下一個節點為共享模式執行緒節點(isShared())就將下一個節點喚醒doReleaseShared();
 private void doReleaseShared() {
        /*
         * Ensure that a release propagates, even if there are other
         * in-progress acquires/releases.  This proceeds in the usual
         * way of trying to unparkSuccessor of head if it needs
         * signal. But if it does not, status is set to PROPAGATE to
         * ensure that upon release, propagation continues.
         * Additionally, we must loop in case a new node is added
         * while we are doing this. Also, unlike other uses of
         * unparkSuccessor, we need to know if CAS to reset status
         * fails, if so rechecking.
         */
        for (;;) {
            Node h = head;
            if (h != null && h != tail) {
                int ws = h.waitStatus;
                if (ws == Node.SIGNAL) {//如果頭節點狀態為signal(-1),即立刻喚醒頭結點後驅節點指向的執行緒。
                    if (!compareAndSetWaitStatus(h, Node.SIGNAL, 0))//如果頭結點的狀設定的原子操作失敗,則重複設定。
                        continue;            // loop to recheck cases
                    unparkSuccessor(h);//喚醒head下個節點的執行緒。
                }
                else if (ws == 0 &&			//如果頭節點狀態為0,即節點初始化狀態,則直接設定頭結點狀態變為PROPAGATE(-3)
                         !compareAndSetWaitStatus(h, 0, Node.PROPAGATE))
                    continue;                // loop on failed CAS
            }
            if (h == head)                   // loop if head changed
                break;//迴圈在此結束,當head頭節點沒有發生變化
        }
    }

共享模式下,所有AQS佇列下的執行緒都會被喚醒。

countDown()方法呼叫的是AQS類的releaseShared(1)方法,方法原始碼如下:

public final boolean releaseShared(int arg)	 {
        if (tryReleaseShared(arg)) {
            doReleaseShared();
            return true;
        }
        return false;
    }
tryReleaseShared(1)返回成功後就執行doReleaseShared方法。tryReleaseShared方法是將state值減掉個1,並賦給它值。如果state值本來就是0,則返回false,只有當

當state減去1後值為0的時候,函式返回true。然後就將AQS中所有的阻塞執行緒都依次喚醒。

總結:到這裡我們可以看到,在這個多執行緒同步模式下,如果你有number個執行緒需要提前執行完成,你就需要建立stae值為number的CountDownLatch類物件c,然後將物件c

的值傳給我們提前執行的執行緒和需要等待執行的執行緒。在這兩中執行緒中分別呼叫c物件的countDown方法和await方法。在此主要實現這個功能的類是AQS類。不過這個同步鎖是一個

共享鎖,而不是以前接觸的獨佔鎖,兩種模式的區別是:共享鎖等待佇列中,只要一個節點獲取了鎖,那麼所有的節點都能獲取鎖,而獨佔模式就不一樣了,一個阻塞節點是要唯一擁有

鎖的。

以上就是對多執行緒同步工具類CountDownLatch的原始碼原理分析。