Rumah >Java >javaTutorial >Bagaimanakah seni bina AQS penyegerak java melepaskan kunci dan menyegerakkan baris gilir

Bagaimanakah seni bina AQS penyegerak java melepaskan kunci dan menyegerakkan baris gilir

WBOY
WBOYke hadapan
2023-05-11 17:16:151564semak imbas

    Pengenalan

    AQS mempunyai terlalu banyak kandungan, jadi kami membahagikannya kepada dua bab Pelajar yang belum membaca separuh pertama AQS boleh melihat ke belakang. Ha, separuh pertama bab ini bercakap tentang banyak konsep asas kunci, sifat asas, cara mendapatkan kunci, dsb. Dalam bab ini kita bercakap tentang cara melepaskan kunci dan baris gilir penyegerakan.

    1. Lepaskan kunci

    Masa pencetus untuk melepaskan kunci ialah kaedah Lock.unLock () yang biasa kami gunakan, tujuannya adalah untuk membenarkan utas melepaskan akses kepada sumber (lihat bahagian laluan ungu dalam gambar rajah seni bina keseluruhan untuk proses ).

    Keluaran kunci juga dibahagikan kepada dua kategori, satu ialah keluaran kunci eksklusif, dan satu lagi ialah keluaran kunci kongsi Mari kita lihat secara berasingan.

    1.1. Lepaskan pelepas kunci eksklusif

    Pelepasan kunci eksklusif agak mudah Mulakan dari kepala barisan dan cari nod seterusnya Ia akan bermula dari penghujung dan mencari nod yang statusnya tidak dibatalkan, dan kemudian melepaskan nod Kod sumber adalah seperti berikut:

    // unlock 的基础方法
    public final boolean release(int arg) {
        // tryRelease 交给实现类去实现,一般就是用当前同步器状态减去 arg,如果返回 true 说明成功释放锁。
        if (tryRelease(arg)) {
            Node h = head;
            // 头节点不为空,并且非初始化状态
            if (h != null && h.waitStatus != 0)
                // 从头开始唤醒等待锁的节点
                unparkSuccessor(h);
            return true;
        }
        return false;
    }
    // 很有意思的方法,当线程释放锁成功后,从 node 开始唤醒同步队列中的节点
    // 通过唤醒机制,保证线程不会一直在同步队列中阻塞等待
    private void unparkSuccessor(Node node) {
        // node 节点是当前释放锁的节点,也是同步队列的头节点
        int ws = node.waitStatus;
        // 如果节点已经被取消了,把节点的状态置为初始化
        if (ws < 0)
            compareAndSetWaitStatus(node, ws, 0);
        // 拿出 node 节点的后面一个节点
        Node s = node.next;
        // s 为空,表示 node 的后一个节点为空
        // s.waitStatus 大于0,代表 s 节点已经被取消了
        // 遇到以上这两种情况,就从队尾开始,向前遍历,找到第一个 waitStatus 字段不是被取消的
        if (s == null || s.waitStatus > 0) {
            s = null;
            // 这里从尾迭代,而不是从头开始迭代是有原因的。
            // 主要是因为节点被阻塞的时候,是在 acquireQueued 方法里面被阻塞的,唤醒时也一定会在 acquireQueued 方法里面被唤醒,唤醒之后的条件是,判断当前节点的前置节点是否是头节点,这里是判断当前节点的前置节点,所以这里必须使用从尾到头的迭代顺序才行,目的就是为了过滤掉无效的前置节点,不然节点被唤醒时,发现其前置节点还是无效节点,就又会陷入阻塞。
            for (Node t = tail; t != null && t != node; t = t.prev)
                // t.waitStatus <= 0 说明 t 没有被取消,肯定还在等待被唤醒
                if (t.waitStatus <= 0)
                    s = t;
        }
        // 唤醒以上代码找到的线程
        if (s != null)
            LockSupport.unpark(s.thread);
    }

    1.2 Lepaskan keluaran kunci kongsiDikongsi

    The kaedah melepaskan kunci kongsi ialah releaseShared, yang kebanyakannya terbahagi kepada Dua langkah:

    cubaReleaseShared untuk melepaskan kunci kongsi semasa, mengembalikan palsu jika gagal dan pergi ke langkah 2 jika berjaya; > membangunkan nod penyekat seterusnya bagi nod semasa Kami telah melihat kaedah ini sebelum ini, dan benang memperoleh bahagian Apabila mengunci, ia akan membangunkan nod di belakangnya. Nama kaedah ialah: doReleaseShared.

    Mari kita lihat kod sumber releaseShared:

    // 共享模式下,释放当前线程的共享锁
    public final boolean releaseShared(int arg) {
        if (tryReleaseShared(arg)) {
            // 这个方法就是线程在获得锁时,唤醒后续节点时调用的方法
            doReleaseShared();
            return true;
        }
        return false;
    }

    2 kaedah penting giliran bersyarat

    Sebelum melihat kaedah giliran bersyarat, kita mesti faham dahulu. mengapa terdapat Jika anda mempunyai baris gilir penyegerakan, adakah anda masih memerlukan baris gilir bersyarat?

    Terutamanya kerana tidak semua senario boleh dikendalikan oleh baris gilir penyegerakan Apabila menghadapi senario di mana kunci + baris gilir digabungkan, Kunci + Keadaan diperlukan Mula-mula menggunakan Kunci untuk menentukan utas yang boleh mendapatkan Kunci, yang diperlukan untuk beratur dan menyekat dalam baris gilir penyegerakan; apabila beberapa utas yang memperoleh kunci menghadapi baris gilir penuh atau kosong, anda boleh menggunakan Syarat untuk menguruskan utas ini, biarkan utas ini menyekat dan menunggu, dan kemudian disekat pada masa yang sesuai. Bangun seperti biasa.

    Senario di mana baris gilir penyegerakan + baris gilir bersyarat digunakan bersama adalah yang paling biasa digunakan dalam senario kunci + baris gilir.

    Jadi giliran bersyarat juga merupakan bahagian yang sangat diperlukan.

    Seterusnya mari kita lihat beberapa kaedah yang lebih penting dalam baris gilir keadaan Kaedah berikut semuanya berada dalam kelas dalaman ConditionObject.

    2.1. Masukkan barisan dan tunggu menunggu

    // 线程入条件队列
    public final void await() throws InterruptedException {
        if (Thread.interrupted())
            throw new InterruptedException();
        // 加入到条件队列的队尾
        Node node = addConditionWaiter();
        // 标记位置 A
        // 加入条件队列后,会释放 lock 时申请的资源,唤醒同步队列队列头的节点
        // 自己马上就要阻塞了,必须马上释放之前 lock 的资源,不然自己不被唤醒的话,别的线程永远得不到该共享资源了
        int savedState = fullyRelease(node);
        int interruptMode = 0;
        // 确认node不在同步队列上,再阻塞,如果 node 在同步队列上,是不能够上锁的
        // 目前想到的只有两种可能:
        // 1:node 刚被加入到条件队列中,立马就被其他线程 signal 转移到同步队列中去了
        // 2:线程之前在条件队列中沉睡,被唤醒后加入到同步队列中去
        while (!isOnSyncQueue(node)) {
            // this = AbstractQueuedSynchronizer$ConditionObject
            // 阻塞在条件队列上
            LockSupport.park(this);
            if ((interruptMode = checkInterruptWhileWaiting(node)) != 0)
                break;
        }
        // 标记位置 B
        // 其他线程通过 signal 已经把 node 从条件队列中转移到同步队列中的数据结构中去了
        // 所以这里节点苏醒了,直接尝试 acquireQueued
        if (acquireQueued(node, savedState) && interruptMode != THROW_IE)
            interruptMode = REINTERRUPT;
        if (node.nextWaiter != null) // clean up if cancelled
            // 如果状态不是CONDITION,就会自动删除
            unlinkCancelledWaiters();
        if (interruptMode != 0)
            reportInterruptAfterWait(interruptMode);
    }

    Terdapat beberapa perkara yang memerlukan perhatian khusus dalam kaedah menunggu:

    Kod di atas menandakan kedudukan A. Sebelum nod sedia untuk memasuki baris gilir bersyarat, ia mesti Kunci yang dipegang pada masa ini akan dilepaskan terlebih dahulu, jika tidak, ia akan memasuki baris gilir keadaan dan benang lain tidak akan dapat mendapatkan kunci di atas menandakan kedudukan B. Pada masa ini, nod dibangkitkan oleh kaedah Condition.signal atau signalAll Pada masa ini, nod telah Ia telah berjaya dipindahkan ke baris gilir penyegerakan (proses biru dalam gambar rajah seni bina keseluruhan), jadi kaedah acquireQueued boleh dilaksanakan secara langsung dalam baris gilir keadaan. Kod sumber suka menggunakan Waiter untuk menamakannya, jadi kita lihat dalam baris gilir syarat Waiter sebenarnya Node.

    Terdapat dua kaedah penting dalam kaedah menunggu: addConditionWaiter dan unlinkCancelledWaiters Mari kita lihat satu persatu.

    2.1.1, addConditionWaiter

    Kaedah addConditionWaiter terutamanya meletakkan nod ke dalam baris gilir keadaan Kod sumber kaedah adalah seperti berikut:
    rreee

    Proses keseluruhan adalah secara relatif. mudah, hanya tambahkan pada baris gilir Pada penghujungnya, terdapat kaedah penting yang dipanggil unlinkCancelledWaiters Kaedah ini akan memadam semua nod dalam baris gilir keadaan yang statusnya bukan SYARAT berikut:

    2.1.2, unlinkCancelledWaiters

    // 增加新的 waiter 到队列中,返回新添加的 waiter
    // 如果尾节点状态不是 CONDITION 状态,删除条件队列中所有状态不是 CONDITION 的节点
    // 如果队列为空,新增节点作为队列头节点,否则追加到尾节点上
    private Node addConditionWaiter() {
        Node t = lastWaiter;
        // If lastWaiter is cancelled, clean out.
        // 如果尾部的 waiter 不是 CONDITION 状态了,删除
        if (t != null && t.waitStatus != Node.CONDITION) {
            unlinkCancelledWaiters();
            t = lastWaiter;
        }
        // 新建条件队列 node
        Node node = new Node(Thread.currentThread(), Node.CONDITION);
        // 队列是空的,直接放到队列头
        if (t == null)
            firstWaiter = node;
        // 队列不为空,直接到队列尾部
        else
            t.nextWaiter = node;
        lastWaiter = node;
        return node;
    }
    Untuk memudahkan semua orang memahami kaedah ini, saya melukis gambar rajah penjelasan, seperti berikut:

    Bagaimanakah seni bina AQS penyegerak java melepaskan kunci dan menyegerakkan baris gilir2.2. Isyarat bangun tunggal

    Kaedah isyarat ialah bangun Bermaksud, contohnya, baris gilir penuh sebelum ini, dan beberapa utas telah disekat dalam baris gilir bersyarat disebabkan oleh mengambil operasi. Tiba-tiba elemen dalam baris gilir dimakan oleh benang A, dan benang A akan memanggil kaedah isyarat untuk membangunkan benang yang disekat sebelum ini dan bermula dari Nod kepala baris bersyarat mula bangun (lihat bahagian biru dalam gambar rajah seni bina keseluruhan untuk proses). Kod sumber adalah seperti berikut:

    // 会检查尾部的 waiter 是不是已经不是CONDITION状态了
    // 如果不是,删除这些 waiter
    private void unlinkCancelledWaiters() {
        Node t = firstWaiter;
        // trail 表示上一个状态,这个字段作用非常大,可以把状态都是 CONDITION 的 node 串联起来,即使 node 之间有其他节点都可以
        Node trail = null;
        while (t != null) {
            Node next = t.nextWaiter;
            // 当前node的状态不是CONDITION,删除自己
            if (t.waitStatus != Node.CONDITION) {
                //删除当前node
                t.nextWaiter = null;
                // 如果 trail 是空的,咱们循环又是从头开始的,说明从头到当前节点的状态都不是 CONDITION
                // 都已经被删除了,所以移动队列头节点到当前节点的下一个节点
                if (trail == null)
                    firstWaiter = next;
                // 如果找到上次状态是CONDITION的节点的话,先把当前节点删掉,然后把自己挂到上一个状态是 CONDITION 的节点上
                else
                    trail.nextWaiter = next;
                // 遍历结束,最后一次找到的CONDITION节点就是尾节点
                if (next == null)
                    lastWaiter = trail;
            }
            // 状态是 CONDITION 的 Node
            else
                trail = t;
            // 继续循环,循环顺序从头到尾
            t = next;
        }
    }

    Mari kita lihat kaedah yang paling kritikal: transferForSignal.

    // 唤醒阻塞在条件队列中的节点
    public final void signal() {
        if (!isHeldExclusively())
            throw new IllegalMonitorStateException();
        // 从头节点开始唤醒
        Node first = firstWaiter;
        if (first != null)
            // doSignal 方法会把条件队列中的节点转移到同步队列中去
            doSignal(first);
    }
    // 把条件队列头节点转移到同步队列去
    private void doSignal(Node first) {
        do {
            // nextWaiter为空,说明到队尾了
            if ( (firstWaiter = first.nextWaiter) == null)
                lastWaiter = null;
            // 从队列头部开始唤醒,所以直接把头节点.next 置为 null,这种操作其实就是把 node 从条件队列中移除了
            // 这里有个重要的点是,每次唤醒都是从队列头部开始唤醒,所以把 next 置为 null 没有关系,如果唤醒是从任意节点开始唤醒的话,就会有问题,容易造成链表的割裂
            first.nextWaiter = null;
            // transferForSignal 方法会把节点转移到同步队列中去
            // 通过 while 保证 transferForSignal 能成功
            // 等待队列的 node 不用管他,在 await 的时候,会自动清除状态不是 Condition 的节点(通过 unlinkCancelledWaiters 方法)
            // (first = firstWaiter) != null  = true 的话,表示还可以继续循环, = false 说明队列中的元素已经循环完了
        } while (!transferForSignal(first) &&
                 (first = firstWaiter) != null);
    }

    Selepas membaca keseluruhan kod sumber, kita dapat melihat bahawa membangkitkan nod dalam baris gilir keadaan sebenarnya memindahkan nod dalam baris gilir syarat ke baris gilir penyegerakan dan menetapkan status nod sebelumnya kepada SIGNAL.

    2.3. Bangun semua isyaratSemua

    signalAll digunakan untuk membangunkan semua nod dalam baris gilir keadaan adalah seperti berikut:

    // 返回 true 表示转移成功, false 失败
    // 大概思路:
    // 1. node 追加到同步队列的队尾
    // 2. 将 node 的前一个节点状态置为 SIGNAL,成功直接返回,失败直接唤醒
    // 可以看出来 node 的状态此时是 0 了
    final boolean transferForSignal(Node node) {
        /*
         * If cannot change waitStatus, the node has been cancelled.
         */
        // 将 node 的状态从 CONDITION 修改成初始化,失败返回 false
        if (!compareAndSetWaitStatus(node, Node.CONDITION, 0))
            return false;
        // 当前队列加入到同步队列,返回的 p 是 node 在同步队列中的前一个节点
        // 看命名是 p,实际是 pre 单词的缩写
        Node p = enq(node);
        int ws = p.waitStatus;
        // 状态修改成 SIGNAL,如果成功直接返回
        // 把当前节点的前一个节点修改成 SIGNAL 的原因,是因为 SIGNAL 本身就表示当前节点后面的节点都是需要被唤醒的
        if (ws > 0 || !compareAndSetWaitStatus(p, ws, Node.SIGNAL))
            // 如果 p 节点被取消,或者状态不能修改成SIGNAL,直接唤醒
            LockSupport.unpark(node.thread);
        return true;
    }

    Seperti yang dapat dilihat dari. kod sumber, intipatinya adalah untuk Kaedah transferForSignal dipanggil dalam gelung untuk memindahkan nod dalam baris gilir keadaan ke baris gilir penyegerakan.

    3. Ringkasan

    Kod sumber AQS akhirnya selesai?

    Bagaimanakah seni bina AQS penyegerak java melepaskan kunci dan menyegerakkan baris gilir

    Atas ialah kandungan terperinci Bagaimanakah seni bina AQS penyegerak java melepaskan kunci dan menyegerakkan baris gilir. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

    Kenyataan:
    Artikel ini dikembalikan pada:yisu.com. Jika ada pelanggaran, sila hubungi admin@php.cn Padam