深入理解Java中的鎖(一)

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

Java中鎖的概念 自旋鎖 : 是指當一個線程在獲取鎖的時候,如果鎖已經被其他線程獲取,那麼該線程將迴圈等待,然後不斷判斷鎖是否能夠被成功獲取,直到獲取到鎖才會退出迴圈。 樂觀鎖 : 假定沒有衝突,在修改數據時如果發現數據和之前獲取的不一致,則讀最新數據,修改後重試修改 悲觀鎖 :假定會發生併發衝突 ...


Java中鎖的概念

自旋鎖 : 是指當一個線程在獲取鎖的時候,如果鎖已經被其他線程獲取,那麼該線程將迴圈等待,然後不斷判斷鎖是否能夠被成功獲取,直到獲取到鎖才會退出迴圈。

樂觀鎖 : 假定沒有衝突,在修改數據時如果發現數據和之前獲取的不一致,則讀最新數據,修改後重試修改

悲觀鎖 :假定會發生併發衝突,同步所有對數據的相關操作,從讀數據就開始上鎖

獨享鎖(寫) : 給資源加上寫鎖,擁有該鎖的線程可以修改資源,其他線程不能再加鎖(單寫)

共用鎖(讀) : 給資源加上讀鎖後只能讀不能改,其他線程也只能加讀鎖,不能加寫鎖 (多讀)

可重入鎖 :線程拿到一把鎖後,可以自由進入同一把鎖所同步的代碼

不可重入鎖 :線程拿到一把鎖後,不可以自由進入同一把鎖所同步的代碼

公平鎖 :爭搶鎖的順序,按照先來後到的順序

非公平鎖 :爭搶鎖的順序,不按照先來後到的順序

Java中幾種重要的鎖實現方式:synchronized, ReentrantLock, ReentrantReadWriteLock

同步關鍵字synchronized

  • 用於實例方法,靜態方法時,隱式指定鎖對象
  • 用於代碼塊時顯示指定鎖對象
  • 鎖的作用域:對象鎖,類鎖,分散式鎖

synchronized特性:可重入,獨享,悲觀鎖
鎖優化:

  • 鎖消除是發生在編譯器級別的一種鎖優化方式,是指虛擬機即時編譯器在運行時,對一些代碼上要求同步,但是被檢測到不可能存在共用數據競爭的鎖進行削除(開啟鎖消除的參數:-xx:+DoEscapeAnalysis -XX:+EliminateLocks)
  • 鎖粗化是指有些情況下我們反而希望把很多次鎖的請求合併成一個請求,以降低短時間內大量鎖請求、同步、釋放帶來的性能損耗

Note: synchronized關鍵字,不僅實現同步,JMM中規定,synchronized要保證可見性(不能夠被緩存)

synchronized用法代碼示例:

public class Counter {

  private static int i = 0;

  // 等價於 synchronized(this)
  public synchronized void update() {
    i++;
  }

  public void updateBlock() {
    synchronized (this) {
      i++;
    }
  }

  // 等價於 synchronized (Counter.class)
  public static synchronized void staticUpdate() {
    i++;
  }

  public static void staticUpdateBlock() {
    synchronized (Counter.class) {
      i++;
    }
  }
}

那麼synchronized加鎖在JVM中到底是如何實現的?

要瞭解synchronized加鎖在JVM中是如何實現的,就有必要瞭解Java對象在JVM中到底是如何存儲的。我們知道JVM中在方法區存儲對象的引用,在堆中存儲的對象實例。那麼堆中存儲的對象又有那些信息哪?其實堆中存儲的對象主要由三部分組成,對象頭,實例欄位數據以及padding。對象頭裡面存儲了指向方法區元數據的引用,實例欄位數據就是存儲了實際的欄位數據,padding主要是為了補位,實例對象在堆中存儲的時候必須是八位元組的整數倍,不夠的時候由padding占位補齊。
img

對象頭中的數據有具體分為Mark World,Class Metadata Address以及Array Length

  • Mark World : 一段32/64的記憶體區域,用來存儲Hashcode、GC分代年齡、鎖狀態標誌、線程持有的鎖、偏向線程ID、偏向時間戳等等
  • Class Metadata Address : 指向類的元信息的引用
  • Array Length : 如果是數組對象,會有一個Array Length用來標記數組的長度

img

輕量級鎖

輕量級鎖的加鎖過程:

  1. 每個線程都會在棧幀中開闢一塊記憶體空間叫 Lock Record
  2. 然後線程會把對象頭中 Mark world 的內容拷貝到 Lock Record
  3. 然後,以拷貝的 Mark world 的 記憶體為舊值,以 Lock Record Address 為新值,通過CAS操作進行搶鎖
  4. 如果Mark world通過CAS操作成功,則成功搶到鎖
  5. 如果CAS操作失敗會進行自旋一定的次數進行搶鎖,如果一定次數還沒搶到則升級為重量級鎖

img

重量級鎖

線程在獲取輕量級鎖失敗的時候會進行自旋,如果不加以限制會對CPU資源造成較多的消耗,所以自旋一定的次數之後會升級成重量級鎖。
我們知道Java中每個對象都會有一個對象監視器(Object Monitor, 即管程),而升級為重量級鎖就需要用到這個Object Monitor。它會有一個owner用來標記這個鎖被誰占用了,還有一個entry list用來存儲未獲得鎖的線程,entry list中的線程都是blocked狀態。假設兩個線程T1,T2同時去獲取重量級鎖,如果T1獲取到了鎖,那麼owner就會指向T1,而T2就會進入entry list進行等待,從而減少對CPU的消耗。

img

偏向鎖

在JDK6以後,預設已經開啟了偏向鎖這個優化,可以通過JVM參數 -XX:-UseBiasedLocking來禁用偏向鎖。若偏向鎖開啟,只有一個線程搶鎖,可獲取偏向鎖。偏向鎖會偏向於第一個獲得它的線程,如果在接下來的執行過程中,該鎖沒有被其他的線程獲取,則持有偏向鎖的線程將永遠不需要同步。大多數情況下,鎖不僅不存在多線程競爭,而且總是由同一線程多次獲得,為了讓線程獲得鎖的代價更低而引入了偏向鎖。當鎖對象第一次被線程獲取的時候,線程使用CAS操作把這個線程的ID記錄在對象Mark Word之中,同時置偏向標誌位1。以後該線程在進入和退出同步塊時不需要進行CAS操作來加鎖和解鎖,只需要簡單地測試一下對象頭的Mark Word里是否存儲著指向當前線程的ID。如果測試成功,表示線程已經獲得了鎖。當有另外一個線程去嘗試獲取這個鎖時,偏向模式就宣告結束。根據鎖對象目前是否處於被鎖定的狀態,撤銷偏向後恢復到未鎖定或輕量級鎖定狀態。

鎖的升級過程

img

 

 

 

 


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

-Advertisement-
Play Games
更多相關文章
  • 五、PHP綜合應用 ftp、ssh、http、telnet、https ftp:File Transfer Protocol,文件傳輸協議,是應用層的協議,它基於傳輸層,為用戶服務,它們負責進行文件的傳輸,其預設埠是21。 ssh:Secure Shell,安全外殼協議,建立在應用層和傳輸層基礎上 ...
  • 基礎題 一、String,StringBuffer, StringBuilder 的區別是什麼?String為什麼是不可變的?1. String是字元串常量,StringBuffer和StringBuilder是字元串變數。StringBuffer是線程安全的,StringBuilder是非線程安全 ...
  • 一、高德軟體有限公司python試題及答案 1. 在python中, list, tuple, dict, set有什麼區別, 主要應用在什麼樣的場景? 定義: list: 鏈表, 有序的項目, 通過索引進行查找, 使用方括弧"[]"; tuple: 元組, 元組將多樣的對象集合到一起, 不能修改, ...
  • 本章內容主要分享多個module中的實體類集合生成到一個jar包中,並且發佈到遠程庫;這裡採用maven-assembly-plugin插件的功能來操作打包,內容不長卻貼近實戰切值得擁有,主要節點內容如: 多個module實體類集合打jar包 jar包打入本地庫 jar包上傳至遠程庫 多個modul ...
  • 位元組碼指令   Java虛擬機的位元組碼指令由一個位元組長度,代表著某種特定操作含義的操作碼以及跟隨其後的零至多個代表此操作所需參數的操作數所構成的。如果忽略異常,JVM的解釋器通過下麵的偽代碼可有效工作: 操作位元組碼   可以利用開源庫直接操作位元組碼,如CGLi ...
  • 手把手教你破解文件密碼、wifi密碼、網頁密碼 1、破解文件密碼: 有時候我們在網上下載一個壓縮包後,必須要關註或者支付一定費用才給你解壓密碼,實屬比較噁心。在這裡手把手叫你實現破解文件解壓密碼。 1、首先我們要導入模塊拿到能解壓的文件包 1 import zipfile 2、拿到你需要解壓的文件( ...
  • 一、 1.連續列印舉例 #打開文件,三個字元一組讀出來內容,然後顯示在屏幕上,每讀一次,停一秒 2.tell函數 (1)用法:用來顯示文件讀寫指針的當前位置 (2)格式:文件.tell() (3)舉例: (4)註意:上面的例子說明瞭:tell返回數字的單位是byte;read是以字元為單位的 3.文 ...
  • 第十章 實戰:ELK日誌分析系統 ElasticSearch、Logstash、Kibana簡稱ELK系統,主要用於日誌的收集與分析。 一個完整的大型分散式系統,會有很多與業務不相關的系統,其中日誌系統是不可或缺的一個,集中式日誌系統需要收集來自不同服務的日誌,對它進行集中管理存儲以及分析。ELK就 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...