Java多線程(6):鎖與AQS(下)

来源:https://www.cnblogs.com/xiangwang1111/archive/2022/10/31/16842277.html
-Advertisement-
Play Games

您好,我是湘王,這是我的博客園,歡迎您來,歡迎您再來~ 之前說過,AQS(抽象隊列同步器)是Java鎖機制的底層實現。既然它這麼優秀,是騾子是馬,就拉出來溜溜吧。 首先用重入鎖來實現簡單的累加,就像這樣: /** * 用重入鎖實現累加 * * @author 湘王 */ public class M ...


您好,我是湘王,這是我的博客園,歡迎您來,歡迎您再來~

 

之前說過,AQS(抽象隊列同步器)Java鎖機制的底層實現。既然它這麼優秀,是騾子是馬,就拉出來溜溜吧。

首先用重入鎖來實現簡單的累加,就像這樣:

/**
 * 用重入鎖實現累加
 *
 * @author 湘王
 */
public class MyLockTest {
    private final Lock lock = new ReentrantLock();
    private int value;
    public int getNext() {
        lock.lock();
        try {
            value++;
        } finally {
            lock.unlock();
        }
        return value;
    }
    public static void main(String[] args) {
        MyLockTest myLock = new MyLockTest();
        for (int i = 0; i < 5; i++) {
            new Thread(new Runnable() {
                @Override
                public void run() {
                    for (int i = 0; i < 5; i++) {
                        System.out.println(myLock.getNext());
                    }
                }
            }).start();
        }
    }
}

 

運行結果顯示數據有重覆:

 

 

 

這麼簡單的計算都能出現重覆,這肯定是無法接受的。

再用獨占鎖來試試看:

/**
 * 利用AQS實現自定義獨占鎖
 *
 * @author 湘王
 */
public class MyExclusiveLock implements Lock {
    @Override
    public void lock() {

    }

    @Override
    public void lockInterruptibly() throws InterruptedException {

    }

    @Override
    public boolean tryLock() {
        return false;
    }

    @Override
    public boolean tryLock(long time, TimeUnit unit) throws InterruptedException {
        return false;
    }

    @Override
    public void unlock() {

    }

    @Override
    public Condition newCondition() {
        return null;
    }
}

 

 

可以看到,實現lock介面,就需要實現若幹自定義的介面。然後以內部類繼承AQS的方式,實現排他鎖,昨天也說過,AQS中tryAcquire()和tryRelease()是一一對應的,也就是也管獲取,一個管釋放,所以代碼是:

/**
 * 內部類繼承AQS的方式,實現排他鎖
 */
private static class SyncHelper extends AbstractQueuedSynchronizer {
    private static final long serialVersionUID = -7666580981453962426L;

    /**
     * 第一個線程進來,拿到鎖就返回true;後面的線程進來,拿不到鎖就返回false
     */
    @Override
    protected boolean tryAcquire(int arg) {
        // 獲取資源狀態
        int state = getState();
        if (0 == state) {// 如果沒有線程拿到資源的鎖
            if (compareAndSetState(0, arg)) {
                // 保存當前持有同步鎖的線程
                setExclusiveOwnerThread(Thread.currentThread());
                return true;
            }
        } else if (Thread.currentThread() == getExclusiveOwnerThread()) {
            // 如果當前線程再次進來,state + 1,可重入
            // 如果這裡沒有這個判斷,那麼程式會卡死
            setState(state + arg);
            return true;
        }
        return false;
    }

    /**
     * 鎖的獲取和釋放需要一一對應
     */
    @Override
    protected boolean tryRelease(int arg) {
        // 獲取資源狀態
        int state = getState();
        // 返回最後一個通過setExclusiveOwnerThread()方法設置過的線程,或者null
        if (Thread.currentThread() != getExclusiveOwnerThread()) {
            throw new RuntimeException();
        }
        setState(state - arg);
        if (0 == state) {
            setExclusiveOwnerThread(null);
            return true;
        }
        return false;
    }

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

 

 

然後再用AQS實現lock介面的方法:

/**
 * 利用AQS實現自定義獨占鎖
 *
 * @author 湘王
 */
public class MyExclusiveLock implements Lock {
    private final SyncHelper synchepler = new SyncHelper();

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

    @Override
    public void lockInterruptibly() throws InterruptedException {
        synchepler.acquireInterruptibly(1);
    }

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

    @Override
    public boolean tryLock(long time, TimeUnit unit) throws InterruptedException {
        return synchepler.tryAcquireNanos(1, unit.toNanos(time));
    }

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

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

    /**
     * 內部類繼承AQS的方式,實現排他鎖
     */
    private static class SyncHelper extends AbstractQueuedSynchronizer {
        private static final long serialVersionUID = -7666580981453962426L;
    
        /**
         * 第一個線程進來,拿到鎖就返回true;後面的線程進來,拿不到鎖就返回false
         */
        @Override
        protected boolean tryAcquire(int arg) {
            // 獲取資源狀態
            int state = getState();
            if (0 == state) {// 如果沒有線程拿到資源的鎖
                if (compareAndSetState(0, arg)) {
                    // 保存當前持有同步鎖的線程
                    setExclusiveOwnerThread(Thread.currentThread());
                    return true;
                }
            } else if (Thread.currentThread() == getExclusiveOwnerThread()) {
                // 如果當前線程再次進來,state + 1,可重入
                // 如果這裡沒有這個判斷,那麼程式會卡死
                setState(state + arg);
                return true;
            }
            return false;
        }
    
        /**
         * 鎖的獲取和釋放需要一一對應
         */
        @Override
        protected boolean tryRelease(int arg) {
            // 獲取資源狀態
            int state = getState();
            // 返回最後一個通過setExclusiveOwnerThread()方法設置過的線程,或者null
            if (Thread.currentThread() != getExclusiveOwnerThread()) {
                throw new RuntimeException();
            }
            setState(state - arg);
            if (0 == state) {
                setExclusiveOwnerThread(null);
                return true;
            }
            return false;
        }
    
        protected Condition newCondition() {
            return new ConditionObject();
        }
    }
}

 

 

然後再運行測試:

/**
 * 實現Lock介面方法並運行排他鎖測試
 *
 * @author 湘王
 */
public class MyExclusiveLockTester {
    // 用自定義AQS獨占鎖實現
    private Lock lock = new MyExclusiveLock();
    private int value;

    public int accmulator() {
        lock.lock();
        try {
            ++value;
        } finally {
            lock.unlock();
        }

        return value;
    }

    public static void main(String[] args) throws InterruptedException {
        MyExclusiveLockTester test = new MyExclusiveLockTester();
        for (int i = 0; i < 5; i++) {
            new Thread(new Runnable() {
                @Override
                public void run() {
                    for (int i = 0; i < 5; i++) {
                        System.out.println(test.accmulator());
                    }
                }
            }).start();
        }
    }
}

 

 

可以看到,結果無論怎麼樣都不會再重覆了。

 

這個只是簡單的累加,接下來用AQS來實現一個實際的生活場景。比如周末帶女票或男票去步行街吃飯,這時候人特別多,需要搖號,而且一次只能進去三張號(不按人頭算,按叫到的號來算),該怎麼實現呢?

可以順著這個思路:搖號機雖有很多號,但它本質上是個共用資源,很多人可以共用,但是每次共用的數量有限。這其實就是個可以指定數量的共用鎖而已。

既然有了思路,那接下來就好辦了。

/**
 * 利用AQS實現自定義共用鎖
 *
 * @author 湘王
 */
public class MyShareLock implements Lock {
    @Override
    public void lock() {
    }

    @Override
    public void lockInterruptibly() throws InterruptedException {
    }

    @Override
    public boolean tryLock() {
        return false;
    }

    @Override
    public boolean tryLock(long time, TimeUnit unit) throws InterruptedException {
        return false;
    }

    @Override
    public void unlock() {
    }

    @Override
    public Condition newCondition() {
        return null;
    }
}

 

 

還是一樣實現Lock介面,但這次是用AQS實現共用鎖。

/**
 * 內部類繼承AQS實現共用鎖
 *
 */
private static class SyncHelper extends AbstractQueuedSynchronizer {
    private static final long serialVersionUID = -7357716912664213942L;

    /**
     * count表示允許幾個線程能同時獲得鎖
     */
    public SyncHelper(int count) {
        if (count <= 0) {
            throw new IllegalArgumentException("鎖資源數量必須大於0");
        }
        // 設置資源總數
        setState(count);
    }

    /**
     * 一次允許多少個線程進來,允許數量的線程都能拿到鎖,其他的線程進入隊列
     */
    @Override
    protected int tryAcquireShared(int acquires) {
        // 自旋
        for (;;) {
            int state = getState();
            int remain = state - acquires;
            // 判斷剩餘鎖資源是否已小於0或者CAS執行是否成功
            if (remain < 0 || compareAndSetState(state, remain)) {
                return remain;
            }
        }
    }

    /**
     * 鎖資源的獲取和釋放要一一對應
     */
    @Override
    protected boolean tryReleaseShared(int releases) {
        // 自旋
        for (;;) {
            // 獲取當前state
            int current = getState();
            // 釋放狀態state增加releases
            int next = current + releases;
            if (next < current) {// 溢出
                throw new Error("Maximum permit count exceeded");
            }
            // 通過CAS更新state的值
            // 這裡不能用setState()
            if (compareAndSetState(current, next)) {
                return true;
            }
        }
    }

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

 

 

然後再來改造之前實現的介面:

/**
 * 利用AQS實現自定義共用鎖
 *
 * @author 湘王
 */
public class MyShareLock implements Lock {
    public static int count;
    private final SyncHelper synchepler = new SyncHelper(count);

    @Override
    public void lock() {
        synchepler.acquireShared(1);
    }

    @Override
    public void lockInterruptibly() throws InterruptedException {
        synchepler.acquireSharedInterruptibly(1);
    }

    @Override
    public boolean tryLock() {
        return synchepler.tryAcquireShared(1) > 0;
    }

    @Override
    public boolean tryLock(long time, TimeUnit unit) throws InterruptedException {
        return synchepler.tryAcquireSharedNanos(1, unit.toNanos(time));
    }

    @Override
    public void unlock() {
        synchepler.releaseShared(1);
    }

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

    /**
     * 內部類繼承AQS實現共用鎖
     *
     */
    private static class SyncHelper extends AbstractQueuedSynchronizer {
        private static final long serialVersionUID = -7357716912664213942L;

        /**
         * count表示允許幾個線程能同時獲得鎖
         */
        public SyncHelper(int count) {
            if (count <= 0) {
                throw new IllegalArgumentException("鎖資源數量必須大於0");
            }
            // 設置資源總數
            setState(count);
        }

        /**
         * 一次允許多少個線程進來,允許數量的線程都能拿到鎖,其他的線程進入隊列
         */
        @Override
        protected int tryAcquireShared(int acquires) {
            // 自旋
            for (;;) {
                int state = getState();
                int remain = state - acquires;
                // 判斷剩餘鎖資源是否已小於0或者CAS執行是否成功
                if (remain < 0 || compareAndSetState(state, remain)) {
                    return remain;
                }
            }
        }

        /**
         * 鎖資源的獲取和釋放要一一對應
         */
        @Override
        protected boolean tryReleaseShared(int releases) {
            // 自旋
            for (;;) {
                // 獲取當前state
                int current = getState();
                // 釋放狀態state增加releases
                int next = current + releases;
                if (next < current) {// 溢出
                    throw new Error("Maximum permit count exceeded");
                }
                // 通過CAS更新state的值
                // 這裡不能用setState()
                if (compareAndSetState(current, next)) {
                    return true;
                }
            }
        }

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

 

 

接下來就該測試咱們需要的效果是否能實現了:

public class MyShareLockTester {
    public static void main(String[] args) throws InterruptedException {
        // 用自定義AQS共用鎖實現
        // 一次允許發放三把鎖
        MyShareLock.count = 3;
        final Lock lock = new MyShareLock();

        // 模擬20個客戶端訪問
        for (int i = 0; i < 20; i++) {
            new Thread(new Runnable() {
                @Override
                public void run() {
                    try {
                        lock.lock();
                        System.out.println("持有 " + Thread.currentThread().getName() + " 的客人可以進餐廳就餐");
                        // 每兩次叫號之間間隔一段時間,模擬真實場景
                        Thread.sleep(3000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    } finally {
                        // 使用完成釋放鎖
                        lock.unlock();
                    }
                }
            }).start();
        }
    }
}

 

這裡有20個號,每次只能發放3張,運行之後就可以看到確實如此。

AQS是個很神奇也很好玩的東西,就像它的作者(也是除了高司令就是對Java影響最大的那個人,整個Java的多線程juc包代碼就是他編寫的Doug LeaAbstractQueuedSynchronizer的註釋中所說:AQS只是一個框架,至於怎麼玩,就是你的事了!

 

 


 

 

感謝您的大駕光臨!咨詢技術、產品、運營和管理相關問題,請關註後留言。歡迎騷擾,不勝榮幸~

 


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

-Advertisement-
Play Games
更多相關文章
  • 一、瀏覽器的渲染過程 註意:這個過程是逐步完成的,為了更好的用戶體驗,渲染引擎將會儘可能早的將內容呈現到屏幕上,並不會等到所有的html 都解析完成之後再去構建和佈局 render 樹。它是解析完一部分內容就顯示一部分內容,同時,可能還在通過網路下載其餘內容。 渲染過程解讀如下: 首先解析收到的文檔 ...
  • 一、函數 數的定義有兩種:一為函數的聲明、二為函數表達式-匿名函數 函數聲明 1 function fn(){//需要函數名 2 console.log("函數聲明"); 3 } 4 fn() 函數表達式 1 const fs=function(){//需要定義變數接收 2 console.log( ...
  • Day1 根據自己的垃圾html+css水平寫出了下麵這個頁面: 首先, 首部的超鏈接都能夠鏈接上, 但是不會[更多]的彈出視窗, 右面的天氣/設置/個人也沒有做, 不會怎麼去調用 中間的圖片和搜索欄能夠正常使用, 搜索欄能夠正常搜索. 但是無法水平垂直居中, 而且, 兩個input無法靠近在一起. ...
  • 相信用過vue的小伙伴,肯定被面試官問過這樣一個問題:在vue中動態的引入圖片為什麼要使用require 有些小伙伴,可能會輕蔑一笑:呵,就這,因為動態添加src被當做靜態資源處理了,沒有進行編譯,所以要加上require, 我倒著都能背出來...... emmm... 乍一看好像說的很有道理啊... ...
  • 通過 JavaScript 在瀏覽器中獲取或設置剪貼板中的內容,常用於一鍵複製或使用網頁油猴複製限制文本 使用 ~~execCommand~~ (已棄用) 寫入文本到剪貼板 document.onclick = function() { let text = 'hello world' let do ...
  • 前言 最近無聊看直播,虎牙廣告是真多,還有一堆ghs直播間經常出現在首頁,不想看到這些直播間,於是想辦法屏蔽直播間。 源碼地址 插件地址 演示 下麵先看看未安裝插件之前 虎牙首頁一堆廣告,視頻自動播放 更過分是一堆ghs,噁心人玩意兒的出現在這裡,真是影響觀看體驗 還有左側推薦欄也很煩 安裝之後: ...
  • 1.4 超鏈接 1.4.1 基礎語法 基礎語法: <a href="網頁地址"> </a> 拓展參數: <a href="網頁地址" target="跳轉方式"> </a> | href | 跳轉鏈接地址 | | | | | target | 鏈接打開方式 | 1.4.2 錨鏈接(id參數) 每個標 ...
  • 為了提高系統吞吐率,也就是提高生產效率,核心觀點如下,系統設計也是如此 在微服務或任何其他基於事件的架構(event-driven-architecture)中,在一些用例中,一個服務可能需要我們對他們自己的本地資料庫進行修改,同時發佈一個事件。然後,該事件會被其他服務所消費。為了擁有一個一致的軟體 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...