分門別類總結Java中的各種鎖,讓你徹底記住

来源:https://www.cnblogs.com/yuxiang1/archive/2019/04/08/10673857.html
-Advertisement-
Play Games

概念 公平鎖/非公平鎖 公平鎖是指多個線程按照申請鎖的順序來獲取鎖。 非公平鎖是指多個線程獲取鎖的順序並不是按照申請鎖的順序,有可能後申請的線程比先申請的線程優先獲取鎖。有可能,會造成優先順序反轉或者饑餓現象。 對於 Java ReentrantLock而言,通過構造函數指定該鎖是否是公平鎖,預設是非 ...


概念

公平鎖/非公平鎖

公平鎖是指多個線程按照申請鎖的順序來獲取鎖。

非公平鎖是指多個線程獲取鎖的順序並不是按照申請鎖的順序,有可能後申請的線程比先申請的線程優先獲取鎖。有可能,會造成優先順序反轉或者饑餓現象。

對於 Java ReentrantLock而言,通過構造函數指定該鎖是否是公平鎖,預設是非公平鎖。非公平鎖的優點在於吞吐量比公平鎖大。

對於Synchronized而言,也是一種非公平鎖。由於其並不像ReentrantLock是通過 AQS 的來實現線程調度,所以並沒有任何辦法使其變成公平鎖。

可重入鎖

可重入鎖又名遞歸鎖,是指在同一個線程在外層方法獲取鎖的時候,在進入內層方法會自動獲取鎖。

說的有點抽象,下麵會有一個代碼的示例。對於 Java ReentrantLock而言, 他的名字就可以看出是一個可重入鎖,其名字是Re entrant Lock重新進入鎖。對於Synchronized而言,也是一個可重入鎖。可重入鎖的一個好處是可一定程度避免死鎖。

synchronized void setA() throws Exception{
    Thread.sleep(1000);
    setB();
}

synchronized void setB() throws Exception{
    Thread.sleep(1000);
}

 

上面的代碼就是一個可重入鎖的一個特點,如果不是可重入鎖的話,setB 可能不會被當前線程執行,可能造成死鎖。

獨享鎖/共用鎖

獨享鎖是指該鎖一次只能被一個線程所持有。

共用鎖是指該鎖可被多個線程所持有。

對於 Java ReentrantLock而言,其是獨享鎖。但是對於 Lock 的另一個實現類ReadWriteLock,其讀鎖是共用鎖,其寫鎖是獨享鎖。讀鎖的共用鎖可保證併發讀是非常高效的,讀寫,寫讀 ,寫寫的過程是互斥的。獨享鎖與共用鎖也是通過 AQS 來實現的,通過實現不同的方法,來實現獨享或者共用。對於Synchronized而言,當然是獨享鎖。

互斥鎖/讀寫鎖

上面講的獨享鎖/共用鎖就是一種廣義的說法,互斥鎖/讀寫鎖就是具體的實現。互斥鎖在 Java 中的具體實現就是ReentrantLock 讀寫鎖在 Java 中的具體實現就是ReadWriteLock

樂觀鎖/悲觀鎖

樂觀鎖與悲觀鎖不是指具體的什麼類型的鎖,而是指看待併發同步的角度。悲觀鎖認為對於同一個數據的併發操作,一定是會發生修改的,哪怕沒有修改,也會認為修改。因此對於同一個數據的併發操作,悲觀鎖採取加鎖的形式。悲觀的認為,不加鎖的併發操作一定會出問題。樂觀鎖則認為對於同一個數據的併發操作,是不會發生修改的。在更新數據的時候,會採用嘗試更新,不斷重新的方式更新數據。樂觀的認為,不加鎖的併發操作是沒有事情的。

從上面的描述我們可以看出,悲觀鎖適合寫操作非常多的場景,樂觀鎖適合讀操作非常多的場景,不加鎖會帶來大量的性能提升。悲觀鎖在 Java 中的使用,就是利用各種鎖。樂觀鎖在 Java 中的使用,是無鎖編程,常常採用的是 CAS 演算法,典型的例子就是原子類,通過 CAS 自旋實現原子操作的更新。

分段鎖

分段鎖其實是一種鎖的設計,並不是具體的一種鎖,對於ConcurrentHashMap而言,其併發的實現就是通過分段鎖的形式來實現高效的併發操作。我們以ConcurrentHashMap來說一下分段鎖的含義以及設計思想,ConcurrentHashMap中的分段鎖稱為 Segment,它即類似於 HashMap(JDK7 與 JDK8 中 HashMap 的實現)的結構,即內部擁有一個 Entry 數組,數組中的每個元素既是一個鏈表;同時又是一個 ReentrantLock(Segment 繼承了 ReentrantLock)。當需要 put 元素的時候,並不是對整個 hashmap 進行加鎖,而是先通過 hashcode 來知道他要放在那一個分段中,然後對這個分段進行加鎖,所以當多線程 put 的時候,只要不是放在一個分段中,就實現了真正的並行的插入。但是,在統計 size 的時候,可就是獲取 hashmap 全局信息的時候,就需要獲取所有的分段鎖才能統計。分段鎖的設計目的是細化鎖的粒度,當操作不需要更新整個數組的時候,就僅僅針對數組中的一項進行加鎖操作。

偏向鎖/輕量級鎖/重量級鎖

這三種鎖是指鎖的狀態,並且是針對Synchronized。在 Java 5 通過引入鎖升級的機制來實現高效Synchronized

這三種鎖的狀態是通過對象監視器在對象頭中的欄位來表明的。

偏向鎖是指一段同步代碼一直被一個線程所訪問,那麼該線程會自動獲取鎖。降低獲取鎖的代價。

輕量級鎖是指當鎖是偏向鎖的時候,被另一個線程所訪問,偏向鎖就會升級為輕量級鎖,其他線程會通過自旋的形式嘗試獲取鎖,不會阻塞,提高性能。

重量級鎖是指當鎖為輕量級鎖的時候,另一個線程雖然是自旋,但自旋不會一直持續下去,當自旋一定次數的時候,還沒有獲取到鎖,就會進入阻塞,該鎖膨脹為重量級鎖。重量級鎖會讓其他申請的線程進入阻塞,性能降低。

自旋鎖

在 Java 中,自旋鎖是指嘗試獲取鎖的線程不會立即阻塞,而是採用迴圈的方式去嘗試獲取鎖,這樣的好處是減少線程上下文切換的消耗,缺點是迴圈會消耗 CPU。

為什麼用 Lock、ReadWriteLock

  • synchronized 的缺陷

    • 被 synchronized 修飾的方法或代碼塊,只能被一個線程訪問。如果這個線程被阻塞,其他線程也只能等待。
    • synchronized 不能響應中斷。
    • synchronized 沒有超時機制。
    • synchronized 只能是非公平鎖。
  • Lock、ReadWriteLock 相較於 synchronized,解決了以上的缺陷:

    • Lock 可以手動釋放鎖(synchronized 獲取鎖和釋放鎖都是自動的),以避免死鎖。
    • Lock 可以響應中斷
    • Lock 可以設置超時時間,避免一致等待
    • Lock 可以選擇公平鎖或非公平鎖兩種模式
    • ReadWriteLock 將讀寫鎖分離,從而使讀寫操作分開,有效提高併發性。

Lock 和 ReentrantLock

要點

如果採用 Lock,必須主動去釋放鎖,並且在發生異常時,不會自動釋放鎖。因此一般來說,使用 Lock 必須在 try catch 塊中進行,並且將釋放鎖的操作放在 finally 塊中進行,以保證鎖一定被被釋放,防止死鎖的發生。

lock() 方法的作用是獲取鎖。如果鎖已被其他線程獲取,則進行等待。

tryLock() 方法的作用是嘗試獲取鎖,如果成功,則返回 true;如果失敗(即鎖已被其他線程獲取),則返回 false。也就是說,這個方法無論如何都會立即返回,獲取不到鎖時不會一直等待。

tryLock(long time, TimeUnit unit) 方法和 tryLock() 方法是類似的,區別僅在於這個方法在獲取不到鎖時會等待一定的時間,在時間期限之內如果還獲取不到鎖,就返回 false。如果如果一開始拿到鎖或者在等待期間內拿到了鎖,則返回 true。

lockInterruptibly() 方法比較特殊,當通過這個方法去獲取鎖時,如果線程正在等待獲取鎖,則這個線程能夠響應中斷,即中斷線程的等待狀態。也就使說,當兩個線程同時通過 lock.lockInterruptibly() 想獲取某個鎖時,假若此時線程 A 獲取到了鎖,而線程 B 只有在等待,那麼對線程 B 調用 threadB.interrupt() 方法能夠中斷線程 B 的等待過程。由於 lockInterruptibly() 的聲明中拋出了異常,所以 lock.lockInterruptibly() 必須放在 try 塊中或者在調用 lockInterruptibly() 的方法外聲明拋出 InterruptedException

註意:當一個線程獲取了鎖之後,是不會被 interrupt() 方法中斷的。因為本身在前面的文章中講過單獨調用 interrupt() 方法不能中斷正在運行過程中的線程,只能中斷阻塞過程中的線程。因此當通過 lockInterruptibly() 方法獲取某個鎖時,如果不能獲取到,只有進行等待的情況下,是可以響應中斷的。

unlock() 方法的作用是釋放鎖。

ReentrantLock 是唯一實現了 Lock 介面的類。

ReentrantLock 字面意為可重入鎖。

源碼

Lock 介面定義

public interface Lock {
    void lock();
    void lockInterruptibly() throws InterruptedException;
    boolean tryLock();
    boolean tryLock(long time, TimeUnit unit) throws InterruptedException;
    void unlock();
    Condition newCondition();
}

 

ReentrantLock 屬性和方法

ReentrantLock 的核心方法當然是 Lock 中的方法(具體實現完全基於 Sync 類中提供的方法)。

此外,ReentrantLock 有兩個構造方法,功能參考下麵源碼片段中的註釋。

// 同步機制完全依賴於此
private final Sync sync;
// 預設初始化 sync 的實例為非公平鎖(NonfairSync)
public ReentrantLock() {}
// 根據 boolean 值選擇初始化 sync 的實例為公平的鎖(FairSync)或不公平鎖(NonfairSync)
public ReentrantLock(boolean fair) {}

Sync

  • Sync 類是 ReentrantLock 的內部類,也是一個抽象類。
  • ReentrantLock 的同步機制幾乎完全依賴於Sync。使用 AQS 狀態來表示鎖的保留數(詳細介紹參見 AQS)。
  • Sync 是一個抽象類,有兩個子類:
    • FairSync - 公平鎖版本。
    • NonfairSync - 非公平鎖版本。

示例

public class ReentrantLockDemo {

    private ArrayList<Integer> arrayList = new ArrayList<Integer>();
    private Lock lock = new ReentrantLock();

    public static void main(String[] args) {
        final ReentrantLockDemo demo = new ReentrantLockDemo();
        new Thread(() -> demo.insert(Thread.currentThread())).start();
        new Thread(() -> demo.insert(Thread.currentThread())).start();
    }

    private void insert(Thread thread) {
        lock.lock();
        try {
            System.out.println(thread.getName() + "得到了鎖");
            for (int i = 0; i < 5; i++) {
                arrayList.add(i);
            }
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            System.out.println(thread.getName() + "釋放了鎖");
            lock.unlock();
        }
    }
}

 


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

-Advertisement-
Play Games
更多相關文章
  • ~~暫時空白....~~ 沒有前置,我用vector存圖 cpp //存儲 struct edge{ int w,to;//w是權值,to是連接到的下一條邊 }; vector e; //連邊 ... for(int i=1;i ...
  • SpringMVC跨域問題排查以及源碼實現 最近一次項目中,將SpringMVC版本從4.1.1升級到4.3.10,出現跨域失敗的情況。關於同源策略和跨域解決方案,網上有很多資料。 項目採用的方式是通過實現過濾器Filter,在Response返回頭文件添加跨域資源共用(CORS) 相關的參數。 發 ...
  • 1.可以把七層協議簡化成四層協議鏈路層 網路層 傳輸層 應用層 2.通過路由器連接的兩個網路網路層ip提供的是一個逐跳協議,提供了一種不可靠的服務,中間有可能會丟傳輸層tcp在ip的基礎上提供了可靠的傳輸層 比喻:tcp就是淘寶賣家 ,ip就是包裹,中間如果包裹丟了,賣家會重新發一個包裹,這裡會有一 ...
  • 鏈表:由一系列不必再記憶體中相連的結構組成,每一個結構均含有表元素和指向後繼結構的指針。 與數組、列表的主要區別: 記憶體不連續; 不能通過下標隨機訪問。 優點: 插入、刪除操作效率高,時間複雜度為o(1); 記憶體利用率高,不會浪費記憶體; 大小不固定,擴展靈活; 缺點: 隨機訪問性差,查找效率低,時間復 ...
  • 死磕 java集合之ConcurrentHashMap源碼分析(一) 它的存儲結構是什麼樣的? 它使用了哪些鎖? 它是怎麼擴容的? 它是否是強一致性的? 它不能解決哪些問題? 它的源碼中使用了哪些不常見的技術? ...
  • '''迭代器:兩個基本方法:iter()和next()迭代器是一個可以記住遍歷的位置的對象。 迭代器對象從集合等第一個元素開始訪問,直到所有的元素被訪問結束,迭代器只能往前不會後退。 迭代器有兩個基本的方法:iter()和next() 字元串,列表或元組對象都可以用於創建迭代器。 迭代器的一大優點是 ...
  • 1. 調用鏈Cat 1.1. 調用鏈演進 1.2. 開源產品比較 1.3. 監控場景 1.4. cat的增值作用 1.5. cat典型報表 1.5.1. 應用報錯大盤 1.5.2. 業務大盤 1.5.3. logView 1.5.4. 可視化的logView 1.5.5. 應用報表(APM) 1.5 ...
  • 一、請求鉤子 在客戶端和伺服器交互的過程中,有些準備工作或稍微工作是需要處理的,比如:在請求開始時,建立資料庫連接;在請求開始時,根據需求進行許可權校驗;在請求結束時,指定數據的交互格式等。為了讓每個視圖函數避免編寫重覆功能的代碼,flask提供了通用設施的功能,即請求鉤子。 例子: 運行結果: 第1 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...