主頁 > 後端開發 > 深入淺出AQS原始碼決議

深入淺出AQS原始碼決議

2020-10-01 08:08:22 後端開發

最近一直在研究AQS的原始碼,希望可以更深刻的理解AQS的實作原理,雖然網上有很多關于AQS的原始碼分析,但是看完以后感徑訓是一知半解,于是,我將自己的整個理解程序記錄下來了,希望對大家有所幫助,

基本原理

AQS是Java中鎖的基礎,主要由兩個佇列組成,一個佇列是同步佇列,另一個是條件佇列

同步佇列的原理

  • 同步佇列的佇列頭部是head,佇列尾部是tail節點,head節點是一個空節點,同步佇列是一個雙向鏈表,通過nextprev連接所有節點
  • 所有的執行緒在競爭鎖的時候都會創建一個Node節點,執行緒與節點系結在一起,(如果是同步鎖和排他鎖不同之處是通過nextWaiter來區分的)并且添加到同步佇列的尾部
  • head的第一個節點獲取鎖,其余節點都需要等待被喚醒
  • 同步佇列中的節點會存在取消和null的情況(如:執行緒超時中斷、執行緒更新節點的中間態),被取消和null的節點不能被喚醒,將會被視為無效節點
  • 一個執行緒只能被有效的前驅節點(取消和null的節點除外)喚醒
  • 持有鎖的執行緒只能是有一個,其他有效節點對應的執行緒都會被掛起

條件佇列的原理

  • 一個同步佇列可以對應多個條件佇列
  • 條件佇列是一個單向鏈表,通過nextWaiter來連接起來,條件佇列的頭節點是firstWaiter,尾節點是lastWaiter
  • 某個條件佇列中滿足條件的節點(被signalsignalAll方法喚醒的節點)才會被轉移到同步佇列
  • 條件佇列中的被轉移到同步佇列的節點是從頭節點開始,條件佇列中被阻塞的執行緒會添加到佇列的尾部

同步佇列的實作

首先,了解以下同步佇列中佇列的節點Node的資料結構

static final class Node {
        /** 共享鎖的標識 */
        static final Node SHARED = new Node();
        /** 排他鎖的標識 */
        static final Node EXCLUSIVE = null;

        /** 執行緒取消 */
        static final int CANCELLED =  1;
        /** 持有鎖的執行緒的后繼執行緒被掛起 */
        static final int SIGNAL    = -1;
        /** 條件佇列標識 */
        static final int CONDITION = -2;
        /**
         * 共享鎖情況下,通知所有其他節點
         */
        static final int PROPAGATE = -3;

        /**
         * waitStatus的取值如下:
         *   SIGNAL(-1): 當前節點的后繼節點應該被掛起
         *   CANCELLED(1): 當前節點被取消
         *   CONDITION(-2): 當前節點在條件佇列
         *   PROPAGATE(-3): 釋放共享鎖時需要通知所有節點
         *   0: 初始值
         *
         */
        volatile int waitStatus;

        /**
         * 前驅節點
         */
        volatile Node prev;

        /**
         * 后繼節點
         */
        volatile Node next;

        /**
         * 節點對應的執行緒
         */
        volatile Thread thread;

        /**
         * 在共享鎖的情況下,該節點的值為SHARED
         * 在排他鎖的情況下,該節點的值為EXCLUSIVE
         * 在條件佇列的情況下,鏈接的是下一個等待條件的執行緒
         */
        Node nextWaiter;
}

其次,我們來看一下同步佇列的鏈表結構
同步佇列鏈表

接著,我們根據同步佇列的原理來分析以下acquirerelease需要做哪些事情:

實作acquire功能需要做的事情

  1. 創建一個Node節點node(該節點可能是排他鎖,也可以能是共享鎖)
  2. node添加到同步佇列尾部,如果同步佇列為空(初始情況下),需要先創建一個空的頭節點,然后再添加到佇列的尾部
  3. 如果node的前驅節點是head,說明node是第一個節點,能夠獲取鎖,需要將head修改成node,釋放前驅節點的資源
  4. 如果node的前驅節點不是head,說明獲取鎖失敗,需要檢測是否需要將node系結的執行緒掛起,分以下幾種情況:
    • 如果nodewaitStatus已經被設定為SIGNAL 表示需要被掛起
    • 如果nodewaitStatus設定為CANCEL表示該節點已經被取消,需要被去掉,并修改 nodeprev,直到鏈接上一個有效的節點為止
    • 否則將nodewaitStatus設定為SIGNAL,表示即將要被掛起
  5. 如果需要將node系結的執行緒掛起,則讓出CPU,直到當前驅節點來喚起node才會開始繼續從步驟3開始執行

與acquire功能相關的代碼

  • acquire方法:獲取排他鎖
public final void acquire(int arg) {
        if (!tryAcquire(arg) &&
            acquireQueued(addWaiter(Node.EXCLUSIVE), arg))
            selfInterrupt();
    }
  1. tryAcquire(arg):對外提供的一個擴展方法,常用的鎖都要實作這個方法,具體實作與鎖相關

  2. addWaiter(Node.EXCLUSIVE): 創建一個排他鎖節點,并將該節點添加到同步佇列尾部,代碼如下:

private Node addWaiter(Node mode) {
        // 創建一個node,EXCLUSIVE型別
        Node node = new Node(mode);

        for (;;) {
            // 獲取尾節點
            Node oldTail = tail;
            if (oldTail != null) {
                // 設定即將成為尾節點的前驅
                node.setPrevRelaxed(oldTail);
                // CAS操作設定尾節點
                if (compareAndSetTail(oldTail, node)) {
                    // 將新尾節點的前驅節點與新的尾節點關聯起來
                    oldTail.next = node;
                    // 回傳添加的節點
                    // 這個節點現在不一定是尾節點,因為如果有多個執行緒呼叫這個方法時,
                    // 可能還有節點添加在這個節點后面
                    return node;
                }
            } else {
                // 如果佇列為空,初始化頭節點
                initializeSyncQueue();
            }
        }
    }
  1. acquireQueued同步佇列中的節點獲取排他鎖
final boolean acquireQueued(final Node node, int arg) {
        try {
            // 執行緒是否中斷
            boolean interrupted = false;
            for (;;) {
                // 獲取前驅節點
                final Node p = node.predecessor();
                // 如果前驅節點是頭節點,獲取鎖
                if (p == head && tryAcquire(arg)) {
                    // 修改頭節點
                    setHead(node);
                    // 釋放頭節點的資源
                    p.next = null; // help GC
                    // 回傳執行緒中斷的狀態
                    // 這也是該方法唯一的回傳值
                    // 沒有獲取鎖的執行緒會一直執行該方法直到獲取鎖以后再回傳
                    return interrupted;
                }
                // 獲取鎖失敗后是否需要將執行緒掛起
                if (shouldParkAfterFailedAcquire(p, node) &&
                    parkAndCheckInterrupt()) // 執行緒掛起并回傳是否被中斷
                    interrupted = true;
            }
        } catch (Throwable t) {
            // 取消該節點
            cancelAcquire(node);
            throw t;
        }
    }
  1. shouldParkAfterFailedAcquire:檢測執行緒獲取鎖失敗以后是否需要被掛起
private static boolean shouldParkAfterFailedAcquire(Node pred, Node node) {
        // 前驅節點的狀態
        int ws = pred.waitStatus;
        if (ws == Node.SIGNAL)
            /*
             * 狀態已經設定成SIGNAL,可以直接掛起該節點
             */
            return true;
        // 節點被取消
        if (ws > 0) {
            /*
             * 找到pred第一個有效的前驅節點
             */
            do {
                node.prev = pred = pred.prev;
            } while (pred.waitStatus > 0);
            // pred可能是一個新的節點,需要將pred的next重寫設定為node
            pred.next = node;
        } else {
            /*
             * CAS操作將pred節點的狀態設定為SIGNAL
             */
            pred.compareAndSetWaitStatus(ws, Node.SIGNAL);
        }
        // 只有當pred節點的waitStatus已經是SIGNAL狀態時,才可以安全的掛起執行緒
        // 否則需要不能被掛起
        return false;
    }
  1. parkAndCheckInterrupt:將當前執行緒掛起,并檢測當前執行緒是否中斷
private final boolean parkAndCheckInterrupt() {
        // 執行緒掛起
        LockSupport.park(this);
        // 檢測執行緒是否中斷
        return Thread.interrupted();
    }
  1. cancelAcquire:取消節點
 private void cancelAcquire(Node node) {
        // 如果節點為空,什么都不做
        if (node == null)
            return;
        // 釋放執行緒
        node.thread = null;

        // 從后往前過濾掉所有的被取消的節點
        Node pred = node.prev;
        while (pred.waitStatus > 0)
            node.prev = pred = pred.prev;

        // 有效前驅節點的nex節點
        Node predNext = pred.next;

        // 將node設定為CANCELLED
        node.waitStatus = Node.CANCELLED;

        // 如果是尾節點,設定新的尾節點
        if (node == tail && compareAndSetTail(node, pred)) {
            // 將新的尾節點的后續設定為null
            pred.compareAndSetNext(predNext, null);
        } else {
            // If successor needs signal, try to set pred's next-link
            // so it will get one. Otherwise wake it up to propagate.
            int ws;
            // 如果前驅節點的執行緒不為null并且waitStatus為SIGNAL
            if (pred != head &&
                ((ws = pred.waitStatus) == Node.SIGNAL ||
                 (ws <= 0 && pred.compareAndSetWaitStatus(ws, Node.SIGNAL))) &&
                pred.thread != null) {
                Node next = node.next;
                // 將node設定成pred的后繼節點
                if (next != null && next.waitStatus <= 0)
                    pred.compareAndSetNext(predNext, next);
            } else {
                // 喚起node節點的后繼節點
                // 因為node節點已經釋放鎖了
                unparkSuccessor(node);
            }

            node.next = node; // help GC
        }
    }
  1. unparkSuccessor:喚醒后繼節點
private void unparkSuccessor(Node node) {
        /*
         * 獲取node節點的waitStatus
         */
        int ws = node.waitStatus;
       // 用CSA操作將waitStatus設定成初始狀態
       // 不管設定是否成功,都無所謂,因為該節點即將被銷毀
        if (ws < 0)
            node.compareAndSetWaitStatus(ws, 0);
        /*
         * 獲取node的后繼節點
         */
        Node s = node.next;
        // 如果后繼節點為null或者被取消,
        // 通過從同步佇列的尾節點開始一直往前找到一個有效的后繼節點
        if (s == null || s.waitStatus > 0) {
            s = null;
            for (Node p = tail; p != node && p != null; p = p.prev)
                if (p.waitStatus <= 0)
                    s = p;
        }
        // 如果后繼節點不為空
        if (s != null)
            LockSupport.unpark(s.thread);// 喚醒后繼節點的執行緒
    }

acquire方法類似的還有acquireInterruptiblytryAcquireNanosacquireSharedacquireSharedInterruptiblytryAcquireSharedNanos,我們都一一分析以下

  • acquireInterruptibly方法:獲取可中斷的排他鎖
public final void acquireInterruptibly(int arg)
            throws InterruptedException {
        if (Thread.interrupted()) // 如果執行緒中斷,直接回傳
            throw new InterruptedException();
        if (!tryAcquire(arg))
            doAcquireInterruptibly(arg); // 中斷式的獲取鎖
    }
  1. doAcquireInterruptibly:可中斷式的獲取鎖
private void doAcquireInterruptibly(int arg)
        throws InterruptedException {
       // 創建一個排他節點加入同步佇列
        final Node node = addWaiter(Node.EXCLUSIVE);
        try {
            for (;;) {
                // 獲取前驅節點
                final Node p = node.predecessor();
                // 如果前驅節點是頭節點,說明已經獲取的鎖
                if (p == head && tryAcquire(arg)) {
                    // 修改頭節點
                    setHead(node);
                    p.next = null; // help GC
                    return;
                }
                // 如果沒有獲取鎖,檢測是否需要掛起
                if (shouldParkAfterFailedAcquire(p, node) &&
                    parkAndCheckInterrupt())
                    throw new InterruptedException(); // 如果發現執行緒已經被中斷,需要拋出例外
            }
        } catch (Throwable t) {
            // 發生例外取消節點
            cancelAcquire(node);
            throw t;
        }
    }
  • tryAcquireNanos方法:超時中斷獲取排他鎖
public final boolean tryAcquireNanos(int arg, long nanosTimeout)
            throws InterruptedException {
        if (Thread.interrupted())
            throw new InterruptedException(); // 執行緒中斷直接回傳
        return tryAcquire(arg) ||
            doAcquireNanos(arg, nanosTimeout); // 超時獲取排他鎖
    }
  1. doAcquireNanos:超時獲取排他鎖
private boolean doAcquireNanos(int arg, long nanosTimeout)
            throws InterruptedException {
        // 如果超時直接回傳
        if (nanosTimeout <= 0L)
            return false;
        // 獲取超時時長
        final long deadline = System.nanoTime() + nanosTimeout;
        // 添加一個排他節點到同步佇列尾部
        final Node node = addWaiter(Node.EXCLUSIVE);
        try {
            for (;;) {
                 // 獲取前驅節點
                final Node p = node.predecessor();
                // 已經獲取鎖
                if (p == head && tryAcquire(arg)) {
                    setHead(node);
                    p.next = null; // help GC
                    return true;
                }
                nanosTimeout = deadline - System.nanoTime();
                // 如果超時了就取消
                if (nanosTimeout <= 0L) {
                    cancelAcquire(node);
                    return false;
                }
                // 檢測節點是否需要被掛起
                if (shouldParkAfterFailedAcquire(p, node) &&
                    nanosTimeout > SPIN_FOR_TIMEOUT_THRESHOLD)
                    // 如果需要掛起,且超時時長大于SPIN_FOR_TIMEOUT_THRESHOLD
                    // 執行緒掛起nanosTimeout時間
                    LockSupport.parkNanos(this, nanosTimeout); 
                if (Thread.interrupted())
                    throw new InterruptedException();
            }
        } catch (Throwable t) {
            // 發生例外取消節點
            cancelAcquire(node);
            throw t;
        }
    }
  • acquireShared方法:獲取共享鎖
public final void acquireShared(int arg) {
        // 對外提供的一個擴展方法,常用的鎖都要實作這個方法,
        // 該方法的實作與鎖的用途有關
        if (tryAcquireShared(arg) < 0) 
            doAcquireShared(arg); // 獲取共享鎖
    }
  1. doAcquireShared:獲取共享鎖
 private void doAcquireShared(int arg) {
        // 添加一個共享節點到同步佇列尾部
        final Node node = addWaiter(Node.SHARED);
        try {
            boolean interrupted = false;
            for (;;) {
                // 獲取前驅節點
                final Node p = node.predecessor();
                if (p == head) {
                    // 回傳結果大于等于0表示獲取共享鎖
                    int r = tryAcquireShared(arg);
                    if (r >= 0) {
                        // 設定頭節點并廣播通知其他獲取共享鎖的節點
                        setHeadAndPropagate(node, r);
                        p.next = null; // help GC
                        // 如果執行緒被中斷,將該執行緒中斷
                        // 共享鎖會被多個執行緒獲取,如果需要中斷
                        // 所有獲取共享鎖的執行緒都要被中斷
                        if (interrupted)
                            selfInterrupt();
                        return;
                    }
                }
                // 檢測是否需要掛起
                if (shouldParkAfterFailedAcquire(p, node) &&
                    parkAndCheckInterrupt()) // 掛起并中斷
                    interrupted = true;
            }
        } catch (Throwable t) {
            // 發生例外取消節點
            cancelAcquire(node);
            throw t;
        }
    }
  1. setHeadAndPropagate:設定頭節點并廣播其他節點來獲取鎖
 private void setHeadAndPropagate(Node node, int propagate) {
        Node h = head; // 記錄舊的頭節點
        setHead(node);// 設定新的頭節點
        /*
         * 如果頭節點為null或者是不是取消狀態,嘗試喚醒后繼節點
         */
        if (propagate > 0 || h == null || h.waitStatus < 0 ||
            (h = head) == null || h.waitStatus < 0) {
            Node s = node.next;
            // node節點的next是SHARED,即共享鎖
            if (s == null || s.isShared())
                // 喚起獲取共享鎖的執行緒
                doReleaseShared();
        }
    }
  1. doReleaseShared:喚醒等待共享鎖的節點
 private void doReleaseShared() {
        /*
         * 喚醒時是從頭節點開始先喚醒第一個共享節點,
         * 第一個共享節點被喚醒后會在doAcquireShared方法里繼續執行(之前就是在這個方法里被掛起的)
         * 第一個共享節點如果獲取鎖會呼叫setHeadAndPropagate方法修改頭節點,然后再呼叫doReleaseShared方法
         * 喚醒第二個共享節點,以此類推,最后把所有的共享節點都喚醒
         */
        for (;;) {
            Node h = head;
            if (h != null && h != tail) {
                // 獲取頭節點的狀態
                int ws = h.waitStatus;
                // 如果頭節點是SIGNAL,需要將狀態設定為0,表示已經即將被喚醒
                if (ws == Node.SIGNAL) {
                    if (!h.compareAndSetWaitStatus(Node.SIGNAL, 0))
                        continue;            // 如果失敗了說明有其他執行緒在修改頭節點,需要繼續重試
                    unparkSuccessor(h); // 喚醒頭節點的后繼節點
                }
                else if (ws == 0 &&
                         !h.compareAndSetWaitStatus(0, Node.PROPAGATE))
                    continue;                // 將頭節點狀態從0設定成PROPAGATE,如果失敗了繼續,因為也有其他獲取共享鎖的執行緒在更改頭節點
            }
            // 如果頭節點未改變(因為沒有后繼節點需要等待共享鎖),跳出回圈
            if (h == head)
                break;
        }
    }
  1. selfInterrupt:中斷當前執行緒
static void selfInterrupt() {
    Thread.currentThread().interrupt();
}
  • acquireSharedInterruptibly方法:可中斷的獲取共享鎖
public final void acquireSharedInterruptibly(int arg)
            throws InterruptedException {
        if (Thread.interrupted())
            throw new InterruptedException(); // 如果執行緒被中斷拋出例外
        if (tryAcquireShared(arg) < 0)
            doAcquireSharedInterruptibly(arg); // 可中斷的方式獲取共享鎖
    }
  1. doAcquireSharedInterruptibly:可中斷的方式后去共享鎖
 private void doAcquireSharedInterruptibly(int arg)
        throws InterruptedException {
        // 添加共享鎖節點到同步佇列尾部
        final Node node = addWaiter(Node.SHARED);
        try {
            for (;;) {
                // 獲取前驅節點
                final Node p = node.predecessor();
                if (p == head) {
                    int r = tryAcquireShared(arg);
                    if (r >= 0) {
                        // 獲取共享鎖以后修改頭節點,通知其他等待共享鎖的節點
                        setHeadAndPropagate(node, r);
                        p.next = null; // help GC
                        return;
                    }
                }
                // 執行緒獲取共享鎖失敗后需要掛起,并且發現執行緒被中斷,所以拋出例外
                if (shouldParkAfterFailedAcquire(p, node) &&
                    parkAndCheckInterrupt())
                    throw new InterruptedException();
            }
        } catch (Throwable t) {
            // 發生例外取消節點
            cancelAcquire(node);
            throw t;
        }
    }
  • tryAcquireSharedNanos方法:超時中斷獲取共享鎖
public final boolean tryAcquireSharedNanos(int arg, long nanosTimeout)
            throws InterruptedException {
        if (Thread.interrupted()) // 執行緒如果中斷了,直接拋出例外
            throw new InterruptedException();
        return tryAcquireShared(arg) >= 0 ||
            doAcquireSharedNanos(arg, nanosTimeout); // 超時獲取共享鎖
    }
  1. doAcquireSharedNanos:超時的方式獲取中斷鎖
private boolean doAcquireSharedNanos(int arg, long nanosTimeout)
            throws InterruptedException {
        // 超時直接回傳
        if (nanosTimeout <= 0L)
            return false;
        final long deadline = System.nanoTime() + nanosTimeout;
        // 添加共享節點到同步佇列尾部
        final Node node = addWaiter(Node.SHARED);
        try {
            for (;;) {
                // 獲取前驅節點
                final Node p = node.predecessor();
                if (p == head) {
                    int r = tryAcquireShared(arg);
                    if (r >= 0) {
                        // 獲取鎖,修改頭節點,通知所有其他等待共享鎖的節點
                        setHeadAndPropagate(node, r);
                        p.next = null; // help GC
                        return true;
                    }
                }
                nanosTimeout = deadline - System.nanoTime();
                if (nanosTimeout <= 0L) {
                    // 超時取消節點
                    cancelAcquire(node);
                    return false;
                }
                if (shouldParkAfterFailedAcquire(p, node) &&
                    nanosTimeout > SPIN_FOR_TIMEOUT_THRESHOLD)
                    // 如果需要掛起,且超時時長大于SPIN_FOR_TIMEOUT_THRESHOLD
                    // 執行緒掛起nanosTimeout時間
                    LockSupport.parkNanos(this, nanosTimeout);
                if (Thread.interrupted())
                    throw new InterruptedException(); // 中斷了拋出例外
            }
        } catch (Throwable t) {
            // 發生例外取消節點
            cancelAcquire(node);
            throw t;
        }
    }

實作release功能需要做的事情

  1. 釋放當前獲取鎖的執行緒持有的資源
  2. 喚醒有效的一個后繼節點

與release功能相關的代碼

  • release方法:釋放排他鎖
    public final boolean release(int arg) {
        if (tryRelease(arg)) {
            Node h = head;
            // 頭節點不能是一個中間態
            if (h != null && h.waitStatus != 0)
                // 喚醒后繼節點
                unparkSuccessor(h);
            return true;
        }
        return false;
    }
  • release方法:釋放共享鎖
public final boolean releaseShared(int arg) {
        if (tryReleaseShared(arg)) {
            // 釋放共享鎖,從頭節點開始一個一個的釋放
            // 如果存在多個共享節點在同步佇列時,doReleaseShared方式其實是遞回呼叫
            doReleaseShared();
            return true;
        }
        return false;
    }

至此,將所有獲取鎖和釋放鎖的方法相關的原始碼全部分析完

條件佇列的實作

我們來看一下條件佇列的鏈表結構
條件佇列的鏈表結構

實作await功能需要做的事情

  1. 創建一個CONDITION型別的節點,將該節點添加到條件佇列
  2. 釋放已經獲取的鎖(因為只有當前執行緒先獲取了鎖才可能再呼叫Condition.await()方法)
  3. 如果無法獲取鎖,執行緒掛起

與await功能相關的代碼

  • await方法:等待條件
public final void await() throws InterruptedException {
            if (Thread.interrupted())
                throw new InterruptedException(); // 如果執行緒中斷,直接拋出例外
            // 創建一個CONDITION型別的節點,將該節點添加到條件佇列尾部
            Node node = addConditionWaiter();
            // 釋放鎖
            // 在呼叫await方法之前都會呼叫lock方法,這個時候已經獲取鎖了
            // 有時候鎖還是可重入的,所以需要將所有的資源都釋放掉
            int savedState = fullyRelease(node);
            int interruptMode = 0;
            // 如果節點不再同步佇列,全部都要掛起
            while (!isOnSyncQueue(node)) {
                LockSupport.park(this);
                // 如果在等待期間發生過中斷(不管是呼叫signal之前還是之后),直接退出
                if ((interruptMode = checkInterruptWhileWaiting(node)) != 0)
                    break;
            }
            // 讓執行緒嘗試去獲取鎖,如果無法獲取鎖就掛起
            if (acquireQueued(node, savedState) && interruptMode != THROW_IE)
                interruptMode = REINTERRUPT;
            // 清除所有在條件佇列中是取消狀態的節點
            if (node.nextWaiter != null) // clean up if cancelled
                unlinkCancelledWaiters();
            // 發生中斷,上報中斷情況
            if (interruptMode != 0)
                reportInterruptAfterWait(interruptMode);
        }
  1. addConditionWaiter:在條件佇列中添加一個節點
 private Node addConditionWaiter() {
            Node t = lastWaiter;
            // 清除條件佇列中無效的節點
            if (t != null && t.waitStatus != Node.CONDITION) {
                unlinkCancelledWaiters();
                t = lastWaiter;
            }
            // 創建一個節點
            Node node = new Node(Node.CONDITION);
            // 添加到條件佇列尾部
            if (t == null)
                firstWaiter = node;
            else
                t.nextWaiter = node;
            lastWaiter = node;
            return node;
        }
  1. unlinkCancelledWaiters:清除在條件佇列中被取消的節點
private void unlinkCancelledWaiters() {
            Node t = firstWaiter;
            Node trail = null;
            // 遍歷條件佇列將所有不是CONDITION狀態的節點全部清除掉
            // 這些節點都是取消狀態的節點
            while (t != null) {
                Node next = t.nextWaiter;
                if (t.waitStatus != Node.CONDITION) {
                    t.nextWaiter = null;
                    if (trail == null)
                        firstWaiter = next;
                    else
                        trail.nextWaiter = next;
                    if (next == null)
                        lastWaiter = trail;
                }
                else
                    trail = t;
                t = next;
            }
        }
  1. fullyRelease:釋放執行緒持有的所有鎖資源
final int fullyRelease(Node node) {
        try {
            int savedState = getState();
            // 釋放所有的資源
            // 如果是可重入鎖,savedState就是重入的次數
            if (release(savedState))
                return savedState;
            throw new IllegalMonitorStateException();
        } catch (Throwable t) {
            // 發生例外就取消該節點
            node.waitStatus = Node.CANCELLED;
            throw t;
        }
    }
  1. isOnSyncQueue:判斷節點是否在同步佇列
final boolean isOnSyncQueue(Node node) {
        // waitStatus是CONDITION或者node沒有前驅節點,說明node不在同步佇列
        if (node.waitStatus == Node.CONDITION || node.prev == null)
            return false;
        if (node.next != null) // 有后繼節點一定在同步隊列
            return true;
        /*
         * 在同步佇列中查找node,看是否在同步佇列中
         */
        return findNodeFromTail(node);
    }
  1. findNodeFromTail:在同步佇列中查找節點
private boolean findNodeFromTail(Node node) {
        // 從尾節點開始查找
        for (Node p = tail;;) {
            if (p == node) // 找到了
                return true;
            if (p == null) // 找到頭了還沒找到
                return false;
            p = p.prev;
        }
    }
  1. checkInterruptWhileWaiting:檢測中斷的情況
private int checkInterruptWhileWaiting(Node node) {
            // 沒有發生中斷回傳0
            // 呼叫signal之前發生中斷回傳THROW_IE
            // 呼叫signal之后發生中斷回傳REINTERRUPT
            return Thread.interrupted() ?
                (transferAfterCancelledWait(node) ? THROW_IE : REINTERRUPT) :
                0;
        }
  1. transferAfterCancelledWait:清除在條件佇列中被取消的節點
// 只有執行緒處于中斷狀態,才會呼叫此方法
// 如果需要的話,將這個已經取消等待的節點轉移到阻塞佇列
// 回傳 true,如果此執行緒在 signal 之前被取消,否則回傳false
final boolean transferAfterCancelledWait(Node node) {
  
        // 用 CAS 將節點狀態設定為 0 
        // 如果這步 CAS 成功,說明是 signal 方法之前發生的中斷,
       // 因為如果 signal 先發生的話,signal 中會將 waitStatus 設定為 0
        if (node.compareAndSetWaitStatus(Node.CONDITION, 0)) {
            enq(node); // 將節點放入阻塞佇列
            return true;
        }
        // 到這里是因為 CAS 失敗,肯定是因為 signal 方法已經將 waitStatus 設定為了 0
        // signal 方法會將節點轉移到阻塞佇列,但是可能還沒完成,這邊自旋等待其完成
        // 當然,這種事情還是比較少的吧:signal 呼叫之后,沒完成轉移之前,發生了中斷
        while (!isOnSyncQueue(node))
            Thread.yield();
        return false;
    }
  1. enq:把節點添加到同步佇列
private Node enq(Node node) {
        // 無限回圈,將節點添加到同步佇列尾部
        for (;;) {
            Node oldTail = tail;
            if (oldTail != null) {
                node.setPrevRelaxed(oldTail);
                if (compareAndSetTail(oldTail, node)) {
                    oldTail.next = node;
                    return oldTail;
                }
            } else {
                // 如果同步佇列為空,初始化
                initializeSyncQueue();
            }
        }
    }
  1. reportInterruptAfterWait:中斷處理
private void reportInterruptAfterWait(int interruptMode)
            throws InterruptedException {
            // 如果是THROW_IE狀態,拋例外
            if (interruptMode == THROW_IE)
                throw new InterruptedException();
            else if (interruptMode == REINTERRUPT) // 再次中斷,因為中斷狀態被使用過一次
                selfInterrupt();
        }

awaitNanosawaitUntilawait(long time, TimeUnit unit)這幾個方法的整體邏輯是一樣的,就不再分析了

實作signal功能需要做的事情

  1. 將條件佇列中的節點加入同步佇列
  2. 喚醒執行緒

與signal功能相關的代碼

  • signal方法:喚醒等待條件的節點
 public final void signal() {
            if (!isHeldExclusively())
                throw new IllegalMonitorStateException();
            // 獲取條件佇列中的第一個節點
            Node first = firstWaiter;
            if (first != null)
                // 喚醒等待條件的節點
                doSignal(first); 
        }
  1. doSignal:喚醒等待條件的節點
private void doSignal(Node first) {
            do {
                // 去掉無效的節點
                if ( (firstWaiter = first.nextWaiter) == null)
                    lastWaiter = null;
                first.nextWaiter = null;
            } while (!transferForSignal(first) &&  // 將節點轉移到同步佇列
                     (first = firstWaiter) != null);
        }
  1. transferForSignal:將節點轉移到同步佇列
final boolean transferForSignal(Node node) {
        /*
         * 取消的節點不需要轉移
         */
        if (!node.compareAndSetWaitStatus(Node.CONDITION, 0))
            return false;

        /*
         * 將節點加入同步佇列尾部
         */
        Node p = enq(node);
        int ws = p.waitStatus;
        // ws > 0 說明 node 在阻塞佇列中的前驅節點取消了等待鎖,直接喚醒 node 對應的執行緒
        // 如果 ws <= 0, 那么 compareAndSetWaitStatus 將會被呼叫
        // 節點入隊后,需要把前驅節點的狀態設為SIGNAL
        if (ws > 0 || !p.compareAndSetWaitStatus(ws, Node.SIGNAL))
            // 如果前驅節點取消或者 CAS 失敗,會進到這里喚醒執行緒
            LockSupport.unpark(node.thread);
        return true;
    }
  • signalAlll方法:喚醒所有等待條件的節點
public final void signalAll() {
            // 如果是當前執行緒
            if (!isHeldExclusively())
                throw new IllegalMonitorStateException();
            Node first = firstWaiter;
            if (first != null)
                // 喚醒所有等待條件的節點
                doSignalAll(first);
        }
  1. doSignalAll:喚醒所有等待條件的節點
// 將所有的節點都轉移到同步佇列
private void doSignalAll(Node first) {
            lastWaiter = firstWaiter = null;
            do {
                Node next = first.nextWaiter;
                first.nextWaiter = null;
                transferForSignal(first);
                first = next;
            } while (first != null);
        }

現在將與AQS相關的核心代碼都整理了一遍,里面如果有描述不清晰或者不準確的地方希望大家可以幫忙指出!

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/144552.html

標籤:Java

上一篇:怎么讀取經緯度資料檔案進行車輛行駛里程的計算

下一篇:有效提高java編程安全性的12潭訓金法則

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • Rust中的智能指標:Box<T> Rc<T> Arc<T> Cell<T> RefCell<T> Weak

    Rust中的智能指標是什么 智能指標(smart pointers)是一類資料結構,是擁有資料所有權和額外功能的指標。是指標的進一步發展 指標(pointer)是一個包含記憶體地址的變數的通用概念。這個地址參考,或 ” 指向”(points at)一些其 他資料 。參考以 & 符號為標志并借用了他們所 ......

    uj5u.com 2023-04-20 07:24:10 more
  • Java的值傳遞和參考傳遞

    值傳遞不會改變本身,參考傳遞(如果傳遞的值需要實體化到堆里)如果發生修改了會改變本身。 1.基本資料型別都是值傳遞 package com.example.basic; public class Test { public static void main(String[] args) { int ......

    uj5u.com 2023-04-20 07:24:04 more
  • [2]SpinalHDL教程——Scala簡單入門

    第一個 Scala 程式 shell里面輸入 $ scala scala> 1 + 1 res0: Int = 2 scala> println("Hello World!") Hello World! 檔案形式 object HelloWorld { /* 這是我的第一個 Scala 程式 * 以 ......

    uj5u.com 2023-04-20 07:23:58 more
  • 理解函式指標和回呼函式

    理解 函式指標 指向函式的指標。比如: 理解函式指標的偽代碼 void (*p)(int type, char *data); // 定義一個函式指標p void func(int type, char *data); // 宣告一個函式func p = func; // 將指標p指向函式func ......

    uj5u.com 2023-04-20 07:23:52 more
  • Django筆記二十五之資料庫函式之日期函式

    本文首發于公眾號:Hunter后端 原文鏈接:Django筆記二十五之資料庫函式之日期函式 日期函式主要介紹兩個大類,Extract() 和 Trunc() Extract() 函式作用是提取日期,比如我們可以提取一個日期欄位的年份,月份,日等資料 Trunc() 的作用則是截取,比如 2022-0 ......

    uj5u.com 2023-04-20 07:23:45 more
  • 一天吃透JVM面試八股文

    什么是JVM? JVM,全稱Java Virtual Machine(Java虛擬機),是通過在實際的計算機上仿真模擬各種計算機功能來實作的。由一套位元組碼指令集、一組暫存器、一個堆疊、一個垃圾回收堆和一個存盤方法域等組成。JVM屏蔽了與作業系統平臺相關的資訊,使得Java程式只需要生成在Java虛擬機 ......

    uj5u.com 2023-04-20 07:23:31 more
  • 使用Java接入小程式訂閱訊息!

    更新完微信服務號的模板訊息之后,我又趕緊把微信小程式的訂閱訊息給實作了!之前我一直以為微信小程式也是要企業才能申請,沒想到小程式個人就能申請。 訊息推送平臺🔥推送下發【郵件】【短信】【微信服務號】【微信小程式】【企業微信】【釘釘】等訊息型別。 https://gitee.com/zhongfuch ......

    uj5u.com 2023-04-20 07:22:59 more
  • java -- 緩沖流、轉換流、序列化流

    緩沖流 緩沖流, 也叫高效流, 按照資料型別分類: 位元組緩沖流:BufferedInputStream,BufferedOutputStream 字符緩沖流:BufferedReader,BufferedWriter 緩沖流的基本原理,是在創建流物件時,會創建一個內置的默認大小的緩沖區陣列,通過緩沖 ......

    uj5u.com 2023-04-20 07:22:49 more
  • Java-SpringBoot-Range請求頭設定實作視頻分段傳輸

    老實說,人太懶了,現在基本都不喜歡寫筆記了,但是網上有關Range請求頭的文章都太水了 下面是抄的一段StackOverflow的代碼...自己大修改過的,寫的注釋挺全的,應該直接看得懂,就不解釋了 寫的不好...只是希望能給視頻網站開發的新手一點點幫助吧. 業務場景:視頻分段傳輸、視頻多段傳輸(理 ......

    uj5u.com 2023-04-20 07:22:42 more
  • Windows 10開發教程_編程入門自學教程_菜鳥教程-免費教程分享

    教程簡介 Windows 10開發入門教程 - 從簡單的步驟了解Windows 10開發,從基本到高級概念,包括簡介,UWP,第一個應用程式,商店,XAML控制元件,資料系結,XAML性能,自適應設計,自適應UI,自適應代碼,檔案管理,SQLite資料庫,應用程式到應用程式通信,應用程式本地化,應用程式 ......

    uj5u.com 2023-04-20 07:22:35 more