乾貨,深入剖析ReentrantLock源碼,推薦收藏

来源:https://www.cnblogs.com/yidengjiagou/archive/2022/11/14/16888174.html
-Advertisement-
Play Games

ReentrantLock和Synchronized都是Java開發中最常用的鎖,與Synchronized這種JVM內置鎖不同的是,ReentrantLock提供了更豐富的語義。可以創建公平鎖或非公平鎖、響應中斷、超時等待、按條件喚醒等。在某些場景下,使用ReentrantLock更適合,功能更強... ...


ReentrantLock和Synchronized都是Java開發中最常用的鎖,與Synchronized這種JVM內置鎖不同的是,ReentrantLock提供了更豐富的語義。可以創建公平鎖或非公平鎖、響應中斷、超時等待、按條件喚醒等。在某些場景下,使用ReentrantLock更適合,功能更強大。

前兩篇文章,我們分析了AQS的加鎖流程、以及源碼實現。當時我們就說了,AQS使用了模板設計模式,父類中定義加鎖流程,子類去實現具體的加鎖邏輯。所以大部分加鎖代碼已經在父類AQS中實現了,導致ReentrantLock的源碼非常簡單,一塊學習一下。

先看一下ReentrantLock怎麼使用?

1. ReentrantLock的使用

/**
 * @author 一燈架構
 * @apiNote ReentrantLock示例
 **/
public class ReentrantLockDemo {
    
    public static void main(String[] args) {
        // 1. 創建ReentrantLock對象
        ReentrantLock lock = new ReentrantLock();
        // 2. 加鎖
        lock.lock();
        try {
            // 3. 這裡執行具體的業務邏輯
        } finally {
            // 4. 釋放鎖
            lock.unlock();
        }
    }
}

可以看到ReentrantLock的使用非常簡單,調用lock加鎖,unlock釋放鎖,需要配置try/finally使用,保證在代碼執行出錯的時候也能釋放鎖。

ReentrantLock也可以配合Condition條件使用,具體可以翻一下前幾篇文章中BlockingQueue的源碼解析,那裡面有ReentrantLock的實際使用。

再看一下ReentrantLock的類結構

2. ReentrantLock類結構

// 實現Lock介面
public class ReentrantLock implements Lock {

    // 只有一個Sync同步變數
    private final Sync sync;

    // Sync繼承自AQS,主要邏輯都在這裡面
    abstract static class Sync extends AbstractQueuedSynchronizer {
    }

    // Sync的兩個子類,分別實現了公平鎖和非公平鎖
    static final class FairSync extends Sync {
    }
    static final class NonfairSync extends Sync {
    }

}

可以看出ReentrantLock的類結構非常簡單,實現了Lock介面。

類裡面有兩個靜態內部類,分別實現公平鎖和非公平鎖。

看一下Lock介面中,定義了哪些方法?

public interface Lock {

    // 加鎖
    void lock();

    // 加可中斷的鎖
    void lockInterruptibly() throws InterruptedException;

    // 嘗試加鎖
    boolean tryLock();

    // 一段時間內,嘗試加鎖
    boolean tryLock(long time, TimeUnit unit) throws InterruptedException;

    // 釋放鎖
    void unlock();

    // 新建條件狀態
    Condition newCondition();
}

就是一些使用鎖的常用方法。

在上篇文章中瀏覽AQS源碼的時候,瞭解到AQS定義了一些有關具體加鎖、釋放鎖的抽象方法,留給子類去實現,再看一下有哪些抽象方法:

// 加獨占鎖
protected boolean tryAcquire(int arg) {
    throw new UnsupportedOperationException();
}
// 釋放獨占鎖
protected boolean tryRelease(int arg) {
    throw new UnsupportedOperationException();
}

// 加共用鎖
protected int tryAcquireShared(int arg) {
    throw new UnsupportedOperationException();
}
// 釋放共用鎖
protected boolean tryReleaseShared(int arg) {
    throw new UnsupportedOperationException();
}

// 判斷是否是當前線程正在持有鎖
protected boolean isHeldExclusively() {
    throw new UnsupportedOperationException();
}

由於ReentrantLock使用的是獨占鎖,所以只需要實現獨占鎖相關的方法就可以了。

3. ReentrantLock源碼解析

3.1 ReentrantLock構造方法

// 預設的構造方法,使用非公平鎖
public ReentrantLock() {
    sync = new NonfairSync();
}

// 傳true,可以指定使用公平鎖
public ReentrantLock(boolean fair) {
    sync = fair ? new FairSync() : new NonfairSync();
}

在創建ReentrantLock對象的時候,可以指定使用公平鎖還是非公平鎖,預設使用非公平鎖,顯然非公平鎖的性能更好。

先思考一個面試常考問題,公平鎖和非公平鎖是怎麼實現的?

3.2 非公平鎖源碼

先看一下加鎖源碼:

從父類ReentrantLock的加鎖方法入口:

public class ReentrantLock implements Lock {
    // 加鎖入口方法
    public void lock() {
        // 調用Sync中加鎖方法
        sync.lock();
    }
}

在子類NonfairSync的加鎖方法:

// 非公平鎖
static final class NonfairSync extends Sync {

    // 加鎖
    final void lock() {
        // 1. 先嘗試加鎖(使用CAS設置state=1)
        if (compareAndSetState(0, 1))
            // 2. 加鎖成功,就把當前線程設置為持有鎖線程
            setExclusiveOwnerThread(Thread.currentThread());
        else
            // 3. 沒加鎖成功,再調用父類AQS中實際的加鎖邏輯
            acquire(1);
    }
}

加鎖邏輯也很簡單,先嘗試使用CAS加鎖(也就是把state從0設置成1),加鎖成功,就把當前線程設置為持有鎖線程。

設計者很聰明,在鎖競爭不激烈的情況下,很大概率可以加鎖成功,也就不用走else中複雜的加鎖邏輯了。

如果沒有加鎖成功,還是需要走else中調用父類AQS的acquire方法,而acquire又需要調用子類的tryAcquire方法。

調用鏈路就是下麵這樣:

image

根據調用鏈路,實際的加鎖邏輯在Sync.nonfairTryAcquire方法裡面。

abstract static class Sync extends AbstractQueuedSynchronizer {
    // 非公平鎖的最終加鎖方法
    final boolean nonfairTryAcquire(int acquires) {
        final Thread current = Thread.currentThread();
        // 1. 獲取同步狀態
        int c = getState();
        // 2. state=0表示無鎖,先嘗試加鎖(使用CAS設置state=1)
        if (c == 0) {
            if (compareAndSetState(0, acquires)) {
                // 3. 加鎖成功,就把當前線程設置為持有鎖線程
                setExclusiveOwnerThread(current);
                return true;
            }
            // 4. 如果當前線程已經持有鎖,執行可重入的邏輯
        } else if (current == getExclusiveOwnerThread()) {
            // 5. 加鎖次數+acquires
            int nextc = c + acquires;
            // 6. 超過tnt類型最大值,溢出了
            if (nextc < 0)
                throw new Error("Maximum lock count exceeded");
            setState(nextc);
            return true;
        }
        return false;
    }
}

再看一下釋放鎖的調用流程,公平鎖和非公平鎖流程是一樣的,最終都是執行Sync.tryRelease方法:

image

abstract static class Sync extends AbstractQueuedSynchronizer {
    // 釋放鎖
    protected final boolean tryRelease(int releases) {
        // 1. 同步狀態減去釋放鎖次數
        int c = getState() - releases;
        // 2. 校驗當前線程不持有鎖,就報錯
        if (Thread.currentThread() != getExclusiveOwnerThread())
            throw new IllegalMonitorStateException();
        boolean free = false;
        // 3. 判斷同步狀態是否等於0,無鎖後,就刪除持有鎖的線程
        if (c == 0) {
            free = true;
            setExclusiveOwnerThread(null);
        }
        setState(c);
        return free;
    }
}

再看一下公平鎖的源碼

3.3 公平鎖源碼

先看一下公平鎖的加鎖流程:

image

最終的加鎖方法是FairSync.tryAcquire,看一下具體邏輯:

static final class FairSync extends Sync {

    // 實現父類的加鎖邏輯
    protected final boolean tryAcquire(int acquires) {
        final Thread current = Thread.currentThread();
        // 1. 獲取同步狀態
        int c = getState();
        // 2. state=0表示無鎖,先嘗試加鎖(使用CAS設置state=1)
        if (c == 0) {
            // 3. 判斷當前線程是不是頭節點的下一個節點(講究先來後到)
            if (!hasQueuedPredecessors() &&
                    compareAndSetState(0, acquires)) {
                setExclusiveOwnerThread(current);
                return true;
            }
            // 4. 如果當前線程已經持有鎖,執行可重入的邏輯
        } else if (current == getExclusiveOwnerThread()) {
            // 5. 加鎖次數+acquires
            int nextc = c + acquires;
            // 6. 超過tnt類型最大值,溢出了
            if (nextc < 0)
                throw new Error("Maximum lock count exceeded");
            setState(nextc);
            return true;
        }
        return false;
    }

    // 判斷當前線程是不是頭節點的下一個節點(講究先來後到)
    public final boolean hasQueuedPredecessors() {
        Node t = tail;
        Node h = head;
        Node s;
        return h != t &&
                ((s = h.next) == null || s.thread != Thread.currentThread());
    }
}

公平鎖的釋放鎖邏輯跟非公平鎖一樣,上面已經講過。

4. 總結

看完了ReentrantLock的所有源碼,是不是覺得ReentrantLock很簡單。

由於加鎖流程的編排工作已經在父類AQS中實現,子類只需要實現具體的加鎖邏輯即可。

加鎖邏輯也很簡單,也就是修改同步狀態state的值和持有鎖的線程exclusiveOwnerThread。

我是「一燈架構」,如果本文對你有幫助,歡迎各位小伙伴點贊、評論和關註,感謝各位老鐵,我們下期見

image


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

-Advertisement-
Play Games
更多相關文章
  • 本篇學習 Yarn Application 編寫方法,將帶你更清楚的瞭解一個任務是如何提交到 Yarn ,在運行中的交互和任務停止的過程。通過瞭解整個任務的運行流程,幫你更好的理解 Yarn 運作方式,出現問題時能更好的定位。 一、簡介 本篇將對 Yarn Application 編寫流程進行介紹。 ...
  • 數據結構是Python中一個很重要的概念,是以某種方式(如通過編號)組合起來的數據元素(如數字、字元乃至其他數據結構)的集合。 在Python中,最基本的數據結構是序列(sequence)。 序列中的每個元素都有編號,及其位置或索引,其中的第一個元素的索引為0,第二個元素位的索引為1,依此類推 在有 ...
  • 先說結論 : extern "C"隻影響到鏈接期的name mangling 什麼是name mangling? 請看 : C++函數重載的實現機制之name mangling - 知乎 (zhihu.com) 舉個例子 : // external.h #ifdef __cplusplus exte ...
  • 迷人的兩度搜索 1、BFS和DFS 深度優先搜索演算法(DFS)和廣度優先搜索演算法(BFS)是一種用於遍歷或搜索樹或圖的演算法,在搜索遍歷的過程中保證每個節點(頂點)訪問一次且僅訪問一次,按照節點(頂點)訪問順序的不同分為深度優先和廣度優先。 1.1、深度優先搜索演算法 深度優先搜索演算法(Depth-Fi ...
  • 今天跟大家分享一個關於“狀態機”的話題。給你講清楚什麼是狀態機、為什麼需要狀態機、適用場景、有哪些具體的實現方案以及各個方案對比(附帶github源碼地址) ...
  • 現代應用已經進入多數據源階段了,不再是一個單一的資料庫包打天下,一個應用中會涉及除關係資料庫外各種數據源,如文本文件類數據、NOSQL、多維資料庫、HTML Webservice等等,即使是關係資料庫,也可能不止一個 應用這樣了,那麼應用中的報表自然也會涉及到多樣性的數據源了 現在的報表,基本都是用 ...
  • 命題邏輯里的一個法則 定義:非p或非q=非(p且q) 最近在看一本書啊《python工匠......》一個騰訊大佬寫的,從這裡面瞭解到這個東西,確實不錯 1 1 # 德摩根定律 2 2 def func(): 3 3 a = 10 4 4 b = 20 5 5 if not a < 5 or not ...
  • 出品 | OSC開源社區(ID:oschina2013) Meta 發佈了一篇博客表示,正在將其 Android 應用的 Java 代碼遷移到 Kotlin,並分享了這一過程中的一些經驗。 該公司認為,Kotlin 是一種流行的 Android 開發語言,與 Java 相比具有一些關鍵優勢。“因此, ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...