深入理解Java中的鎖(三)

来源:https://www.cnblogs.com/coding-diary/archive/2019/07/28/11261238.html
-Advertisement-
Play Games

ReadWriteLock介面 讀寫鎖維護一對關聯鎖,一個只用於讀操作,一個只用於寫操作。讀鎖可以由多個線程同時持有,又稱共用鎖。寫鎖同一時間只能由一個線程持有,又稱互斥鎖。同一時間,兩把鎖不能被不同線程持有。讀寫鎖適合讀取操作多於寫入操作的場景,改進互斥鎖的性能,比如集合的併發安全性改造,緩存組件 ...


ReadWriteLock介面

讀寫鎖維護一對關聯鎖,一個只用於讀操作,一個只用於寫操作。讀鎖可以由多個線程同時持有,又稱共用鎖。寫鎖同一時間只能由一個線程持有,又稱互斥鎖。同一時間,兩把鎖不能被不同線程持有。讀寫鎖適合讀取操作多於寫入操作的場景,改進互斥鎖的性能,比如集合的併發安全性改造,緩存組件等。

ReentrantReadWriteLock實現原理分析

  1. ReentrantReadWriteLock需要一個owner用來標記那個寫操作的線程獲取到了鎖,owner只會標記寫操作的線程引用,不會標記讀操作的線程,一個writeCount用來記錄寫操作加鎖的次數, 一個readCount用來記錄讀操作加鎖的次數,還有一個waiters等待隊列用來存放沒有搶到鎖的線程列表
  2. 當有寫操作線程進來時,會先判斷readCount的值,如果readCount為0說明讀鎖未被占用
  3. 然後判斷writeCount的值,如果writeCount為0,說明寫鎖未被占用
  4. 然後通過CAS操作進行搶鎖將writeCount值加1,如果搶到鎖則將owner設置為當前寫操作線程的引用
  5. 如果writeCount不為0同時owner指向當前寫線程的引用,則將writeCount的值加1
  6. 如果writeCount不為0同時owner指向的不是當前寫線程的引用,則將則將線程放入等待隊列
  7. 如果CAS搶鎖失敗,則將線程放入等待隊列
  8. 如果寫操作線程進來時,readCount不為0說明讀鎖已被占用,則將線程放入等待隊列
  9. 當有讀操作線程進來時,會先判斷writeCount的值,如果writeCount為0說明寫鎖未被占用
  10. 然後通過CAS將readCount的值加1
  11. 如果讀操作線程進來時,writeCount不為0說明寫鎖被占用
  12. 如果寫鎖是被當前線程占用則該線程可以繼續獲得讀鎖,即鎖降級
  13. 如果寫鎖不是被當前線程占用,則將線程放入等待隊列
  14. 當有寫線程釋放鎖時,會將writeCount的值減1,如果writeCount的值為0,則將owner設為null同時喚醒等待隊列頭部的線程出隊列進行搶鎖操作
  15. 如果等待隊列的頭部線程是讀操作,則會進行CAS操作將readCount值加1同時喚醒下一個等待線程
  16. 如果下一個線程還是讀操作,則會進行CAS操作將readCount值加1並且繼續喚醒下一個等待線程
  17. 如果下一個線程是寫操作,則不會喚醒需要等到將讀鎖釋放完之後才會喚醒

手動實現ReentrantReadWriteLock示例:

public class MyReadWriteLock {
  private AtomicInteger readCount = new AtomicInteger(0);
  private AtomicInteger writeCount = new AtomicInteger(0);

  // 獨占鎖 擁有者
  private AtomicReference<Thread> owner = new AtomicReference<>();

  // 等待隊列
  private volatile LinkedBlockingQueue<WaitNode> waiters = new LinkedBlockingQueue<WaitNode>();

  class WaitNode {
    int type = 0; // 0 為想獲取獨占鎖的線程,  1為想獲取共用鎖的線程
    Thread thread = null;
    int arg = 0;

    public WaitNode(Thread thread, int type, int arg) {
      this.thread = thread;
      this.type = type;
      this.arg = arg;
    }
  }

  // 獲取獨占鎖
  public void lockWrite() {
    int arg = 1;
    // 嘗試獲取獨占鎖,若成功,退出方法,    若失敗...
    if (!tryLockWrite(arg)) {
      // 標記為獨占鎖
      WaitNode waitNode = new WaitNode(Thread.currentThread(), 0, arg);
      waiters.offer(waitNode); // 進入等待隊列

      // 迴圈嘗試拿鎖
      for (; ; ) {
        // 若隊列頭部是當前線程
        WaitNode head = waiters.peek();
        if (head != null && head.thread == Thread.currentThread()) {
          if (!tryLockWrite(arg)) { // 再次嘗試獲取 獨占鎖
            LockSupport.park(); // 若失敗,掛起線程
          } else { // 若成功獲取
            waiters.poll(); //  將當前線程從隊列頭部移除
            return; // 並退出方法
          }
        } else { // 若不是隊列頭部元素
          LockSupport.park(); // 將當前線程掛起
        }
      }
    }
  }

  // 釋放獨占鎖
  public boolean unlockWrite() {
    int arg = 1;

    // 嘗試釋放獨占鎖 若失敗返回true,若失敗...
    if (tryUnlockWrite(arg)) {
      WaitNode next = waiters.peek(); // 取出隊列頭部的元素
      if (next != null) {
        Thread th = next.thread;
        LockSupport.unpark(th); // 喚醒隊列頭部的線程
      }
      return true; // 返回true
    }
    return false;
  }

  // 嘗試獲取獨占鎖
  public boolean tryLockWrite(int acquires) {
    // 如果read count !=0 返回false
    if (readCount.get() != 0) return false;

    int wct = writeCount.get(); // 拿到 獨占鎖 當前狀態

    if (wct == 0) {
      if (writeCount.compareAndSet(wct, wct + acquires)) { // 通過修改state來搶鎖
        owner.set(Thread.currentThread()); //  搶到鎖後,直接修改owner為當前線程
        return true;
      }
    } else if (owner.get() == Thread.currentThread()) {
      writeCount.set(wct + acquires); // 修改count值
      return true;
    }

    return false;
  }

  // 嘗試釋放獨占鎖
  public boolean tryUnlockWrite(int releases) {
    // 若當前線程沒有 持有獨占鎖
    if (owner.get() != Thread.currentThread()) {
      throw new IllegalMonitorStateException(); // 拋IllegalMonitorStateException
    }

    int wc = writeCount.get();
    int nextc = wc - releases; // 計算 獨占鎖剩餘占用
    writeCount.set(nextc); // 不管是否完全釋放,都更新count值

    if (nextc == 0) { // 是否完全釋放
      owner.compareAndSet(Thread.currentThread(), null);
      return true;
    } else {
      return false;
    }
  }

  // 獲取共用鎖
  public void lockRead() {
    int arg = 1;

    if (tryLockRead(arg) < 0) { // 如果tryAcquireShare失敗
      // 將當前進程放入隊列
      WaitNode node = new WaitNode(Thread.currentThread(), 1, arg);
      waiters.offer(node); // 加入隊列

      for (; ; ) {
        // 若隊列頭部的元素是當前線程
        WaitNode head = waiters.peek();
        if (head != null && head.thread == Thread.currentThread()) {
          if (tryLockRead(arg) >= 0) { // 嘗試獲取共用鎖,  若成功
            waiters.poll(); // 將當前線程從隊列中移除

            WaitNode next = waiters.peek();
            if (next != null && next.type == 1) { // 如果下一個線程也是等待共用鎖
              LockSupport.unpark(next.thread); // 將其喚醒
            }
            return; // 退出方法
          } else { // 若嘗試失敗
            LockSupport.park(); // 掛起線程
          }
        } else { // 若不是頭部元素
          LockSupport.park();
        }
      }
    }
  }

  // 解鎖共用鎖
  public boolean unLockRead() {
    int arg = 1;

    if (tryUnLockRead(arg)) { // 當read count變為0,才叫release share成功
      WaitNode next = waiters.peek();
      if (next != null) {
        LockSupport.unpark(next.thread);
      }
      return true;
    }
    return false;
  }

  // 嘗試獲取共用鎖
  public int tryLockRead(int acquires) {
    for (; ; ) {
      if (writeCount.get() != 0 && owner.get() != Thread.currentThread()) return -1;

      int rct = readCount.get();
      if (readCount.compareAndSet(rct, rct + acquires)) {
        return 1;
      }
    }
  }

  // 嘗試解鎖共用鎖
  public boolean tryUnLockRead(int releases) {
    for (; ; ) {
      int rc = readCount.get();
      int nextc = rc - releases;
      if (readCount.compareAndSet(rc, nextc)) {
        return nextc == 0;
      }
    }
  }
}

鎖降級

鎖降級指的是寫鎖降級為讀鎖,是指持有寫鎖的同時,再獲取讀鎖,隨後釋放寫鎖的過程。
寫鎖是線程獨占,讀鎖是線程共用,所以寫鎖降級為讀鎖可行,而讀鎖升級為寫鎖不可行。

代碼示例:

class TeacherInfoCache {
  static volatile boolean cacheValid;
  static final ReadWriteLock rwl = new ReentrantReadWriteLock();

  static Object get(String dataKey) {
    Object data = null;

    // 讀取數據,加讀鎖
    rwl.readLock().lock();
    try {
      if (cacheValid) {
        data = Redis.data.get(dataKey);
      } else {
        // 通過加鎖的方式去訪問DB,加寫鎖
        rwl.readLock().unlock();

        rwl.writeLock().lock();
        try {
          if (!cacheValid) {
            data = DataBase.queryUserInfo();
            Redis.data.put(dataKey, data);

            cacheValid = true;
          }
        } finally {
          // 鎖降級
          rwl.readLock().lock();
          rwl.writeLock().unlock();
        }
      }
      return data;
    } finally {
      rwl.readLock().unlock();
    }
  }
}

class DataBase {
  static String queryUserInfo() {
    System.out.println("查詢資料庫。。。");
    return "name:Kody,age:40,gender:true,";
  }
}

class Redis {
  static Map<String, Object> data = new HashMap<>();
}


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

-Advertisement-
Play Games
更多相關文章
  • Java8 增加了 Lambda 表達式,很大程度使代碼變的更加簡潔緊湊了,那麼 Java8 是如何實現 Lambda 表達式的呢? 直接看一個簡單的創建線程的例子。 執行 編譯生成文件 ,然後用 命令來分析這個class文件。 執行 顯示所有類和成員。 由上面的代碼可以看出編譯器根據 Lambda ...
  • 一、原型模式簡介 1、基礎概念 原型模式屬於對象的創建模式。通過給出一個原型對象來指明所有創建的對象的類型,然後用複製這個原型對象的辦法創建出更多同類型的對象。 2、模式結構 原型模式要求對象實現一個可以“克隆”自身的介面,這樣就可以通過複製一個實例對象本身來創建一個新的實例。這樣一來,通過原型實例 ...
  • https://www.cnblogs.com/fireflyupup/p/4875130.html Collection List 在Collection的基礎上引入了有序的概念,位置精確;允許相同元素。在列表上迭代通常優於索引遍歷。特殊的ListIterator迭代器允許元素插入、替換,雙向訪問 ...
  • 一個可以沉迷於技術的程式猿,wx加入加入技術群:fsx641385712 ...
  • 1.字元串的定義 可以使用""雙引號,也可以使用''單引號定義字元串,一般使用雙引號定義。 2.字元串的操作 判斷類型: 查找和替換 大小寫切換: 文本對齊 註:string.center(weight,str) 以str填充對齊,其他兩個方法類似,都可以拓展。 去除空白字元 拆分和鏈接 3.字元串 ...
  • 新聞 "Fantomas 3.0" "宣告.NET Core 3.0預覽版7" ".NET Core 3.0預覽版7中ASP.NET Core與Blazor的升級" "Visual Studio 2019版本16.2正式版本與16.3預覽版1" "Mac上的Visual Studio 2019版本8 ...
  • 1.數組: java.lang.ArrayIndexOutOfBoundsException:5 下標越界異常 java.lang.NullPointerException 空指針異常 arr.length獲取數組長度 數組存儲的是多個數,數據的操作離不開迴圈2數組初始化:int[] arr=new ...
  • 一、Gateway 和 Zuul 的區別 Zuul 基於servlet 2.5 (works with 3.x),使用阻塞API。它不支持任何長期的連接,如websocket。 Gateway建立在Spring Framework 5,Project Reactor 和Spring Boot 2 上 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...