探索抽象同步隊列 AQS

来源:https://www.cnblogs.com/emanjusaka/archive/2023/09/25/page_8.html
-Advertisement-
Play Games

AbstractQueuedSynchronizer抽象同步隊列簡稱AQS,它是實現同步器的基礎組件,併發包中鎖的底層就是使用AQS實現的。大多數開發者可能永遠不會直接使用AQS,但是知道其原理對於架構設計還是很有幫助的。 ...


by emanjusaka from https://www.emanjusaka.top/archives/8 彼岸花開可奈何
本文歡迎分享與聚合,全文轉載請留下原文地址。

前言

AbstractQueuedSynchronizer抽象同步隊列簡稱AQS,它是實現同步器的基礎組件,併發包中鎖的底層就是使用AQS實現的。大多數開發者可能永遠不會直接使用AQS,但是知道其原理對於架構設計還是很有幫助的。AQS是Java中的一個抽象類,全稱是AbstractQueuedSynchronizer,即抽象隊列同步器。它定義了兩種資源共用模式:獨占式和共用式。獨占式每次只能有一個線程持有鎖,例如ReentrantLock實現的就是獨占式的鎖資源;共用式允許多個線程同時獲取鎖,併發訪問共用資源,ReentrantReadWriteLock和CountDownLatch等就是實現的這種模式。AQS維護了一個volatile的state變數和一個FIFO(先進先出)的隊列。其中state變數代表的是競爭資源標識,而隊列代表的是競爭資源失敗的線程排隊時存放的容器 。

一、原理

AQS的核心思想是通過一個FIFO的隊列來管理線程的等待和喚醒,同時維護了一個state變數來表示同步狀態,可以通過getState、setState、compareAndSetState函數修改其值。當一個線程想要獲取鎖時,如果state為0,則表示該線程獲取鎖成功,否則表示該線程獲取鎖失敗。它將被放入等待隊列中,直到滿足特定條件才能再次嘗試獲取。當一個線程釋放鎖時,如果state為1,則表示該線程釋放鎖成功,否則表示該線程釋放鎖失敗。AQS通過CAS操作來實現加鎖和解鎖。

1.1 CLH隊列

image

AQS中的CLH隊列鎖是CLH鎖的一種變體,將自旋操作改成了阻塞線程操作。AQS 中的對 CLH 鎖數據結構的改進主要包括三方面:擴展每個節點的狀態、顯式的維護前驅節點和後繼節點以及諸如出隊節點顯式設為 null 等輔助 GC 的優化。

在 AQS(AbstractQueuedSynchronizer)中使用的 CLH 隊列,head 指針和 tail 指針分別指向 CLH 隊列中的兩個關鍵節點。

  1. head 指針:head 指針指向 CLH 隊列中的首個節點,該節點表示當前持有鎖的線程。當一個線程成功地獲取到鎖時,它就成為了持有鎖的線程,並且會將該信息記錄在 head 指針所指向的節點中。
  2. tail 指針:tail 指針指向 CLH 隊列中的最後一個節點,該節點表示隊列中最後一個等待獲取鎖的線程。當一個線程嘗試獲取鎖時,它會生成一個新的節點,並將其插入到 CLH 隊列的尾部,然後成為 tail 指針所指向的節點。這樣,tail 指針的作用是標記當前 CLH 隊列中最後一個等待獲取鎖的線程。

通過 head 指針和 tail 指針,CLH 隊列能夠維護一種有序的等待隊列結構,保證線程獲取鎖的順序和互斥訪問的正確性。當一個線程釋放鎖時,它會修改當前節點的狀態,並喚醒後繼節點上的線程,讓後續的線程能夠及時感知鎖的釋放,並爭奪獲取鎖的機會。

1.2 線程同步

對於AQS來說,線程同步的關鍵是對狀態值state進行操作。state為0時表示沒有線程持有鎖,大於0時表示有線程持有鎖。根據state是否屬於一個線程,操作state的方式分為獨占方式和共用方式。

在獨占方式下獲取和釋放資源使用的方法為:

  • void acquire(int arg)
  • void acquireInterruptibly(int arg)
  • boolean release(int arg)

使用獨占方式獲取的資源是與具體線程綁定的,就是說如果一個線程獲取到了資源,就會標記是這個線程獲取到了,其他線程再嘗試操作state獲取資源時會發現當前該資源不是自己持有的,就會在獲取失敗後被阻塞。

在共用方式下獲取和釋放資源的方法為:

  • void acquireShared(int arg)
  • voidacquireSharedInterruptibly(int arg)
  • boolean releaseShared(int arg)。

對應共用方式的資源與具體線程是不相關的,當多個線程去請求資源時通過CAS方式競爭獲取資源,當一個線程獲取到了資源後,另外一個線程再次去獲取時如果當前資源還能滿足它的需要,則當前線程只需要使用CAS方式進行獲取即可。

二、資源獲取與釋放

2.1 獨占式

  1. 當一個線程調用acquire(int arg)方法獲取獨占資源時,會首先使用tryAcquire方法嘗試獲取資源,具體是設置狀態變數state的值,成功則直接返回,失敗則將當前線程封裝為類型為Node.EXCLUSIVE的Node節點後插入到AQS阻塞隊列的尾部,並調用LockSupport.park(this)方法掛起自己。

        public final void acquire(int arg) {
          if (! tryAcquire(arg) &&
              acquireQueued(addWaiter(Node.EXCLUSIVE), arg))
              selfInterrupt();
       }
    
  1. 當一個線程調用release(int arg)方法時會嘗試使用tryRelease操作釋放資源,這裡是設置狀態變數state的值,然後調用LockSupport.unpark(thread)方法激活AQS隊列裡面被阻塞的一個線程(thread)。被激活的線程則使用tryAcquire嘗試,看當前狀態變數state的值是否能滿足自己的需要,滿足則該線程被激活,然後繼續向下運行,否則還是會被放入AQS隊列並被掛起。

        public final boolean release(int arg) {
            if (tryRelease(arg)) {
                Node h = head;
                if (h ! = null && h.waitStatus ! = 0)
                    unparkSuccessor(h);
                return true;
            }
            return false;
        }
    

2.2 共用式

  1. 當線程調用acquireShared(int arg)獲取共用資源時,會首先使用tryAcquireShared嘗試獲取資源,具體是設置狀態變數state的值,成功則直接返回,失敗則將當前線程封裝為類型為Node.SHARED的Node節點後插入到AQS阻塞隊列的尾部,並使用LockSupport.park(this)方法掛起自己。

        public final void acquireShared(int arg) {
            if (tryAcquireShared(arg) < 0)
                doAcquireShared(arg);
        }
    
  1. 當一個線程調用releaseShared(int arg)時會嘗試使用tryReleaseShared操作釋放資源,這裡是設置狀態變數state的值,然後使用LockSupport.unpark(thread)激活AQS隊列裡面被阻塞的一個線程(thread)。被激活的線程則使用tryReleaseShared查看當前狀態變數state的值是否能滿足自己的需要,滿足則該線程被激活,然後繼續向下運行,否則還是會被放入AQS隊列並被掛起。

        public final boolean releaseShared(int arg) {
              if (tryReleaseShared(arg)) {
                  doReleaseShared();
                  return true;
              }
              return false;
          }
    

三、基於AQS實現自定義同步器

基於AQS實現一個不可重入的獨占鎖,自定義AQS需要重寫一系列函數,還需要定義原子變數state的含義。這裡定義,state為0表示目前鎖沒有被線程持有,state為1表示鎖已經被某一個線程持有,由於是不可重入鎖,所以不需要記錄持有鎖的線程獲取鎖的次數。

package top.emanjusaka;

import java.io.Serializable;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.AbstractQueuedSynchronizer;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;

public class UnReentrantLock implements Lock, Serializable {
    // 藉助 AbstractQueuedSynchronizer 實現
    private static class Sync extends AbstractQueuedSynchronizer {
        // 查看是否有線程持有鎖
        protected boolean isHeldExclusively() {
            return getState() == 1;
        }
        // 嘗試獲取鎖
        public boolean tryAcquire(int acquires) {
            assert acquires == 1;
            // 使用CAS 設置state
            if (compareAndSetState(0, 1)) {
                // 如果 CAS 操作成功,表示成功獲得了鎖。這時,通過 setExclusiveOwnerThread 方法將當前線程設置為獨占鎖的擁有者
                setExclusiveOwnerThread(Thread.currentThread());
                return true;
            }
            // 如果 CAS 操作失敗,即無法將 state 的值從 0 設置為 1,表示鎖已被其他線程占用,無法獲取鎖,於是返回 false。
            return false;
        }
        // 嘗試釋放鎖
        protected boolean tryRelease(int releases) {
            assert releases == 1;
            if (getState() == 0)
                throw new IllegalMonitorStateException();
            // 釋放成功,將獨占鎖的擁有者設為null
            setExclusiveOwnerThread(null);
            // 將state的值設為0
            setState(0);
            return true;
        }

        Condition newCondition() {
            return new ConditionObject();
        }
    }

    private final Sync sync = new Sync();

    @Override
    public void lock() {
        sync.acquire(1);
    }

    @Override
    public boolean tryLock() {
        return sync.tryAcquire(1);
    }

    @Override
    public void unlock() {
        sync.release(1);
    }

    @Override
    public Condition newCondition() {
        return sync.newCondition();
    }

    public boolean isLocked() {
        return sync.isHeldExclusively();
    }

    public void lockInterruptibly() throws InterruptedException {
        sync.acquireInterruptibly(1);
    }

    public boolean tryLock(long timeout, TimeUnit unit) throws InterruptedException {
        return sync.tryAcquireSharedNanos(1, unit.toNanos(timeout));
    }

}

在釋放鎖時並沒有使用 CAS(Compare and Swap)操作,而是直接使用了 setState​ 方法來將 state​ 的值設置為 0。

這是因為在釋放鎖的過程中,並不需要涉及到多線程併發的問題。只有持有鎖的線程才能夠釋放鎖,其他線程無法對鎖進行操作。因此,不需要使用 CAS 來進行原子性的狀態更新。

在這種情況下,可以直接使用普通的方法來設置 state​ 的值為 0,將獨占鎖的擁有者設為 null。因為只有一個線程可以操作這個鎖,不存在併發競爭的情況,也就不需要使用 CAS 來保證原子性。

需要註意的是,當調用 tryRelease​ 方法時,應該保證當前線程是持有鎖的線程,否則會拋出 IllegalMonitorStateException​ 異常。這是為了確保只有擁有鎖的線程才能釋放鎖,防止誤釋放其他線程的鎖。

四、參考資料

  1. 《併發編程之美》
  2. ​AbstractQueuedSynchronizer​​抽象類的源碼

本文原創,才疏學淺,如有紕漏,歡迎指正。尊貴的朋友,如果本文對您有所幫助,歡迎點贊,並期待您的反饋,以便於不斷優化。
原文地址: https://www.emanjusaka.top/archives/8
微信公眾號:emanjusaka的編程棧


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 來源:麥叔編程 作者:小K 前言 一個好的變數名能讓讀代碼的人(包括寫的人),身心舒暢,但一個“奇葩”的變數名可能會逼瘋一個程式員。 今天是奇葩變數名大賞! 正文 註:以下素材均採集自網路 先上場的是某企業機房的門牌: 我猜這個主任可能是個胡建人。 推薦一個開源免費的 Spring Boot 實戰項 ...
  • 一 背景 C端服務應用升級和重啟,導致耗時瞬時抖動,業務超時,應用監控報警,上游感知明顯,導致用戶體驗變差。 二 應用升級重啟導致抖動的原因 1 C端服務應用升級和重啟的冷啟動階段,它需要重新載入和初始化各種資源,例如資料庫連接、緩存數據等,導致耗時瞬時飆升。 2 應用重啟後,本地緩存失效,應用需要 ...
  • 1 全新併發編程模式 JDK9 後的版本你覺得沒必要折騰,我也認可,但是JDK21有必要關註。因為 JDK21 引入全新的併發編程模式。 一直沽名釣譽的GoLang吹得最厲害的就是協程了。JDK21 中就在這方面做了很大的改進,讓Java併發編程變得更簡單一點,更絲滑一點。 之前寫過JDK21 Fe ...
  • 今天不知為何開始報錯 Entry WEB-INF/classes/classpath.index is a duplicate but no duplicate handling strategy has been set.,大約是由於 我把 Gradle 遷移到了 Kotlin 導致的 經過一番搜 ...
  • 大家好,我是TJ君! 如今在國內運營的各種互聯網應用都有接入IP來源顯示的要求,現在相關API的供應商也很多。今天TJ剛好看到一個不錯的,所以馬上給大家推薦一下。 這款不錯的產品名稱為:IPInfo 產品特性 該IP查詢工具除了傳統的提供地址位置之外,還有很多其他能力,具體的這裡TJ君給大家整理了一 ...
  • 前兩天一個小伙伴突然找我求助,說準備換個坑,最近在系統複習多線程知識,但遇到了一個刷新認知的問題…… ...
  • 大家好,我是mep。今天一起來探討一下Redis緩存的問題,SpringBoot如何集成Redis網上文章很多,基本都是介紹如何配置redisTemplate,如何調用,本文就不過多介紹了。這次我們研究的是:Redis的事務。 首先拋出一個問題,Redis支持事務嗎? 答案肯定是支持,不然也不需要我 ...
  • 如果分析的數據與地域相關,那麼,把分析結果結合地圖一起展示的話,會讓可視化的效果得到極大的提升。 比如,分析各省GDP數據,人口數據,用柱狀圖,餅圖之類的雖然都可以展示分析結果,不過,如果能在全國的地圖上展示各省的分析結果的話,會讓人留下更加深刻的印象。 將數據的分析結果展示在地圖上,難點在於: 如 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...