7.ReadWriteLock介面及其實現ReentrantReadWriteLock

来源:http://www.cnblogs.com/yulinfeng/archive/2017/06/04/6942264.html
-Advertisement-
Play Games

Java併發包的locks包里的鎖基本上已經介紹得差不多了,ReentrantLock重入鎖是個關鍵,在清楚的瞭解了同步器AQS的運行機制後,實際上再分析這些鎖就會顯得容易得多,這章節主講另外一個重要的鎖——ReentrantReadWriteLock讀寫鎖。 ReentrantLock是一個獨占鎖 ...


  Java併發包的locks包里的鎖基本上已經介紹得差不多了,ReentrantLock重入鎖是個關鍵,在清楚的瞭解了同步器AQS的運行機制後,實際上再分析這些鎖就會顯得容易得多,這章節主講另外一個重要的鎖——ReentrantReadWriteLock讀寫鎖。

  ReentrantLock是一個獨占鎖,也就是說只能由一個線程獲取鎖,但如果場景是線程只做讀的操作呢?這樣ReentrantLock就不是很合適,讀的線程並不需要保證其線程的安全性,任何一個線程都能去獲取鎖,只有這樣才能儘可能地保證性能和效率。ReentrantReadWriteLock就是這樣的一個鎖,在其內部分為讀鎖和寫鎖,可以有N個讀操作線程獲取到寫鎖,但是只能有1個寫操作線程獲取到寫鎖,那麼可以預見的是寫鎖是共用鎖(AQS中的共用模式),讀鎖是獨占鎖(AQS中的獨占模式)。首先來看讀寫鎖的介面類:

1 public interface ReadWriteLock {    
2     Lock readLock();        //獲取讀鎖
3     Lock writeLock();        //獲取寫鎖
4 }

  可以看到ReadWriteLock介面只定義了兩個方法,獲取讀鎖和獲取寫鎖的方法。下麵是ReadWriteLock的實現類——ReentrantReadWriteLock。  

  和ReentrantLock類似,ReentrantReadWriteLock在其內部也是通過一個內部類Sync實現同步器AQS,同樣也是通過實現Sync實現公平鎖和非公平鎖,這一點的思路和ReentrantLock類似。在ReadWriteLock介面中獲取的讀鎖和寫鎖是怎麼實現的呢?

//ReentrantReadWriteLock
private final ReentrantReadWriteLock.ReadLock readerLock;
private final ReentrantReadWriteLock.WriteLock writerLock;
final Sync sync;
public ReentrantReadWriteLock(){
    this(false);    //預設非公平鎖
}
public ReentrantReadWriteLock(boolean fair) {
    sync = fair ? new FairSync() : new NonfairSync();    //鎖類型(公平/非公平)
    readerLock = new ReadLock(this);    //構造讀鎖
    writerLock = new WriteLock(this);    //構造寫鎖
}
……
public ReentrantReadWriteLock.WriteLock writeLock0{return writerLock;}
public ReentrantReadWriteLock.ReadLock readLock0{return ReaderLock;}
//ReentrantReadWriteLock$ReadLock
public static class ReadLock implements Lock {
    protected ReadLock(ReentrantReadwritLock lock) {
        sync = lock.sync;        //最後還是通過Sync內部類實現鎖
  }
    ……    //它實現的是Lock介面,其餘的實現可以和ReentrantLock作對比,獲取鎖、釋放鎖等等
}
//ReentrantReadWriteLock$WriteLock
public static class WriteLock implemnts Lock {
    protected WriteLock(ReentrantReadWriteLock lock) {
        sync = lock.sync;
  }
……    //它實現的是Lock介面,其餘的實現可以和ReentrantLock作對比,獲取鎖、釋放鎖等等
}

  上面是對ReentrantReadWriteLock做了一個大致的介紹,可以看到在其內部有好幾個內部類,實際上讀寫鎖內有兩個鎖——ReadLock、WriteLock,這兩個鎖都是實現自Lock介面,可以和ReentrantLock對比,而這兩個鎖的內部實現則是通過Sync,也就是同步器AQS實現的,這也可以和ReentrantLock中的Sync對比。
  回顧一下AQS,其內部有兩個重要的數據結構——一個是同步隊列、一個則是同步狀態,這個同步狀態應用到讀寫鎖中也就是讀寫狀態,但AQS中只有一個state整型來表示同步狀態,讀寫鎖中則有讀、寫兩個同步狀態需要記錄。所以,讀寫鎖將AQS中的state整型做了一下處理,它是一個int型變數一共4個位元組32位,那麼可以讀寫狀態就可以各占16位——高16位表示讀,低16位表示寫。

  

  現在有一個疑問如果state的值位5,二進位為(00000000000000000000000000000101),如何快速確定讀和寫各自的狀態呢?這就要用到位移運算了。計算方式為:寫狀態state & 0x0000FFFF,讀狀態state >>> 16。寫狀態增加1等於state + 1,讀狀態增加1等於state + (1 << 16)。有關移位運算可以參考《<<、>>、>>>移位操作》

寫鎖的獲取與釋放

  根據我們之前的經驗可以得知:AQS已經將獲取鎖的演算法骨架搭好了,只需子類實現tryAcquire(獨占鎖),故我們只需查看tryAcquire。

 1 //ReentrantReadWriteLock$Sync
 2 protected final boolean tryAcquire(int acquires) {
 3     Thread current = Thread.currentThread;
 4     int c = getState();    //獲取state狀態
 5     int w = exclusiveCount(c);    //獲取寫狀態,即 state & 0x00001111
 6     if (c != 0) {    //存在同步狀態(讀或寫),作下一步判斷
 7         if (w == 0 || current != getExclusiveOwnerThread())     //寫狀態為0,但同步狀態不為0表示有讀狀態,此時獲取鎖失敗,或者當前已經有其他寫線程獲取了鎖此時也獲取鎖失敗
 8             return false;
 9         if (w + exclusiveCount(acquire) > MAX_COUNT)    //鎖重入是否超過限制
10             throw new Error(“Maxium lock count exceeded”);
11         setState(c + acquire);    //記錄鎖狀態
12         return true;
13   }
14   if (writerShouldBlock() || !compareAndSetState(c, c + acquires))
15       return false;        //writerShouldBlock對於非公平鎖總是返回false,對於公平鎖則判斷同步隊列中是否有前驅節點
16   setExclusiveOwnerThread(current);
17   return true;
18 }

  上面是寫鎖的狀態獲取,不好理解的是writerShouldBlock方法,此方法上面有描述,非公平鎖直接返回false,而對於公平鎖則是調用hasQueuedPredecessors方法如下:

1 //ReentrantReadWriteLock$FairSync
2 final boolean writerShouldBlock() {
3     return hasQueuedPredecessors();
4 }

  原因是為什麼呢?這就要回到非公平鎖和公平鎖的區別上來了,簡單回顧一下,詳情可參考《5.Lock介面及其實現ReentrantLock》。對於非公平鎖,每次線程獲取鎖時首先會強行進行鎖獲取操作而不管同步隊列中是否有線程,當獲取不到時才會將線程構造至隊尾;對於公平鎖來講,只要同步隊列中存線上程,就不會去獲取鎖,而是將線程構造添加至隊尾。所以重新回到寫狀態的獲取上,tryAcquire方法里,前面發現沒有線程持有鎖,但是此時會根據鎖的不同做相應操作,對於非公平鎖——搶鎖,對公平鎖——同步隊列中有線程,不搶鎖,添加至隊尾排隊。
  寫鎖的釋放與ReentrantLock的釋放過程基本類似,畢竟都是獨占鎖,每次釋放減少寫的狀態,直到減小到0就表示寫鎖已經完全釋放。

讀鎖的獲取與釋放

  同理,根據我們之前的經驗可以得知:AQS已經將獲取鎖的演算法骨架搭好了,只需子類實現tryAcquireShared(共用鎖),故我們只需查看tryAcquireShared。我們知道對於共用模式下的鎖,它能夠被多個線程同時獲取,現在問題來了,T1線程獲取了鎖,同步狀態state=1,此時T2也獲取了鎖,state=2,接著T1線程重入state=3,也就是說讀狀態是所有線程讀鎖次數的總和,而每個線程各自獲取讀鎖的次數只能選擇保存在ThreadLock中,由線程自身維護,所以在這個地方要做一些複雜處理,源碼有點長,但複雜就在於每個線程保存自身獲取讀鎖的次數,具體參照源碼的tryAcquireShared,仔細閱讀並結合上面對寫鎖獲取的分析不難讀懂。
  讀鎖的釋放值得註意的地方在於自身維護的獲取鎖的次數,以及通過移位操作減少狀態state – (1 << 16)。

 


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

-Advertisement-
Play Games
更多相關文章
  • 序列標註(sequence labelling),輸入序列每一幀預測一個類別。OCR(Optical Character Recognition 光學字元識別)。 MIT口語系統研究組Rob Kassel收集,斯坦福大學人工智慧實驗室Ben Taskar預處理OCR數據集(http://ai.sta ...
  • 一開始,我得向Libuv庫和Libuv庫開發者以及相關粉絲們道一個歉,對不起,我錯怪你們了。深深感到自己的無知,是多麼羞愧的事情!! 事情的經過是這樣的。 原先按照公司要求,我在開發Windows版的TCP伺服器時,使用了Libuv庫。正是因為Libuv庫的強大,才讓我們老大推薦使用。我們老大學識淵 ...
  • 作業二:多級菜單 (1)三級菜單 (2)可以次選擇進入各子菜單 (3)所需新知識點:列表、字典 要求:輸入back返回上一層,輸入quit退出整個程式 思路: (1)首先定義好三級菜單字典; (2)提取第一級省的編號,列印包含哪些省份,讓用戶輸入省份的編號,能夠顯示對應的省,這個過程需要創建一個字典 ...
  • 今天我做JUnit關於MySQL測試時發現,類似於assertNull(tu)之類的代碼不知道什麼意思,因此稍微總結如下。 org.springframework.util.AssertAssert翻譯為中文為"斷言".大概來說,就是斷定某一個實際的值就為自己預期想得到的,如果不一樣就拋出異常. s ...
  • 環境配置1:安裝mysql,環境變數添加mysql的bin目錄 環境配置2:python安裝MySQL-Python 請根據自身操作系統下載安裝,否則會報c ++ compile 9.0,import _mysql等錯誤 windows10 64位操作系統可到 http://www.lfd.uci. ...
  • 今日學習:hibernate是什麼 一、hibernate是什麼 框架是什麼: 1.框架是用來提高開發效率的 2.封裝了好了一些功能.我們需要使用這些功能時,調用即可.不需要再手動實現. 3.所以框架可以理解成是一個半成品的項目.只要懂得如何駕馭這些功能即可. hibernate框架是什麼: hib ...
  • 資料 "A literature review of UAV 3D path planning" 上面那個論文把uav的路徑規劃分為以下5類: sampling based algorithms node based algorithms mathematical model based algor ...
  • 閑來無事上知乎,看到好多妹子,於是抓取一波。 有沒有興趣?? 目標網址https://www.zhihu.com/collection/78172986 抓取分析 爬取分析 使用pandas操作文件 那麼,下一步就是對名字進行分詞了,jieba分詞,你值得擁有。fxsjy/jieba 下一步就是分詞 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...