深入理解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
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...