詳細聊聊 Synchronized,以及鎖的升級過程

在Java中,synchronized關鍵字是用于實現線程同步的重要機制,它通過內置鎖(Monitor)確保多個線程對共享資源的安全訪問。


1. synchronized 的基本使用與實現原理

使用方式
  • 修飾實例方法:鎖是當前對象實例。
    public synchronized void method() { ... }
    
  • 修飾靜態方法:鎖是當前類的Class對象。
    public static synchronized void staticMethod() { ... }
    
  • 同步代碼塊:需顯式指定鎖對象(任意對象均可)。
    synchronized (lockObject) { ... }
    
底層實現
  • Monitor 機制:每個對象關聯一個Monitor(監視器鎖),通過monitorentermonitorexit字節碼指令實現加鎖/解鎖。
    • 線程進入同步塊時,執行monitorenter嘗試獲取鎖。
    • 退出同步塊時,執行monitorexit釋放鎖。

2. 對象頭與鎖狀態標記

每個Java對象在內存中分為三部分:對象頭(Header)實例數據(Instance Data)對齊填充(Padding)。對象頭中的Mark Word字段記錄了鎖狀態信息。

Mark Word 結構(以64位JVM為例)
鎖狀態存儲內容
無鎖對象的哈希碼、分代年齡、是否偏向鎖(1 bit)
偏向鎖偏向線程ID、偏向時間戳、分代年齡、鎖標志位(01)
輕量級鎖指向線程棧中鎖記錄(Lock Record)的指針,鎖標志位(00)
重量級鎖指向Monitor對象(重量級鎖)的指針,鎖標志位(10)

3. 鎖的升級過程

JVM根據線程競爭情況動態調整鎖狀態,以減少性能開銷。鎖升級的路徑為:
無鎖 → 偏向鎖 → 輕量級鎖 → 重量級鎖

(1) 偏向鎖(Biased Locking)
  • 適用場景:單線程反復進入同步塊,無實際競爭。
  • 核心機制
    • 對象首次被線程訪問時,將線程ID寫入Mark Word,進入偏向模式。
    • 后續該線程進入同步塊時,無需執行CAS操作,直接檢查線程ID是否匹配。
  • 優勢:消除無競爭時的同步開銷。
  • 撤銷條件
    • 其他線程嘗試獲取鎖時,觸發偏向鎖撤銷。
    • 需要等待全局安全點(STW),檢查原線程是否存活或已釋放鎖。
(2) 輕量級鎖(Lightweight Locking)
  • 適用場景:多線程交替執行同步塊,競爭不激烈。
  • 核心機制
    • 線程在棧幀中創建鎖記錄(Lock Record),將對象Mark Word復制到鎖記錄中。
    • 通過CAS將Mark Word替換為指向鎖記錄的指針。成功則獲取鎖;失敗則膨脹為重量級鎖。
  • 優勢:避免線程阻塞,通過自旋(CAS)減少內核態切換開銷。
  • 自旋優化
    • 適應性自旋:JVM根據歷史自旋成功率動態調整自旋次數。
(3) 重量級鎖(Heavyweight Locking)
  • 適用場景:多線程高并發競爭。
  • 核心機制
    • Monitor對象(C++實現)管理線程競爭,包含_owner(持有者)、_EntryList(阻塞隊列)、_WaitSet(等待隊列)。
    • 未獲取鎖的線程進入_EntryList,由操作系統調度(涉及用戶態到內核態切換)。
  • 特點:線程阻塞,響應慢但公平。

4. 鎖升級的觸發條件

步驟觸發條件
無鎖 → 偏向鎖對象首次被線程訪問,JVM啟用偏向鎖(默認開啟,Java 15后需手動開啟)。
偏向鎖 → 輕量級鎖其他線程嘗試獲取鎖,導致偏向鎖撤銷。
輕量級鎖 → 重量級鎖CAS自旋失敗(超過閾值或競爭激烈),觸發鎖膨脹(Inflate)。

5. 鎖的不可逆性與性能權衡

  • 不可逆性:鎖升級后無法降級,因為降級會增加復雜性和性能損耗。
  • 性能權衡
    • 偏向鎖:適合單線程場景,但撤銷成本高(需STW)。
    • 輕量級鎖:適合低競爭場景,依賴CAS自旋。
    • 重量級鎖:適合高競爭場景,犧牲響應時間保證穩定性。

6. Monitor 的詳細結構

Monitor對象(如ObjectMonitor)包含以下關鍵字段:

  • _owner:當前持有鎖的線程。
  • _recursions:鎖的重入次數。
  • _EntryList:等待獲取鎖的線程隊列。
  • _WaitSet:調用wait()后進入等待狀態的線程隊列。

7. 實際案例:鎖升級過程

場景:兩個線程交替執行同步塊
  1. 初始狀態:對象無鎖。
  2. 線程A進入同步塊:升級為偏向鎖,Mark Word記錄線程A的ID。
  3. 線程B嘗試進入:觸發偏向鎖撤銷,升級為輕量級鎖,線程B通過CAS競爭。
  4. 線程B CAS失敗:自旋后仍未成功,升級為重量級鎖,線程B進入_EntryList阻塞。

8. 最佳實踐

  • 避免過度同步:減少鎖粒度(如使用ConcurrentHashMap)。
  • 優先使用輕量級工具:如ReentrantLockStampedLock(需手動管理)。
  • 監控鎖競爭:通過JVM參數(-XX:+PrintFlagsFinal)或工具(Arthas)分析鎖狀態。

在這里插入圖片描述

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/bicheng/80223.shtml
繁體地址,請注明出處:http://hk.pswp.cn/bicheng/80223.shtml
英文地址,請注明出處:http://en.pswp.cn/bicheng/80223.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

vue3的深入組件-組件 v-model

組件 v-model 基本用法? v-model 可以在組件上使用以實現雙向綁定。 從 Vue 3.4 開始&#xff0c;推薦的實現方式是使用 defineModel() 宏&#xff1a; <script setup> const model defineModel()function update() {model.value } </script><template>…

15.thinkphp的上傳功能

一&#xff0e;上傳功能 1. 如果要實現上傳功能&#xff0c;首先需要建立一個上傳表單&#xff0c;具體如下&#xff1a; <form action"http://localhost/tp6/public/upload"enctype"multipart/form-data" method"post"><input type&…

word文檔基本操作: 編輯頁眉頁腳和插入目錄

文章目錄 引言I 編輯頁眉頁腳II 插入目錄III 知識擴展基于axure畫架構圖基于Knife4j導出接口文檔基于PDManer導出數據庫設計文檔引言 背景: 信息安全認證需要準備相關文件用于審核 一般的開發設計包含總體設計、概要設計、詳細設計、接口設計、數據庫設計、部署結構設計、原型…

Qt 通過控件按鈕實現hello world + 命名規范(7)

文章目錄 使用編輯框來完成 hello world通過編輯圖形化界面方式通過純代碼方式 通過按鈕的方式來創建 hello world通過編輯圖形化界面方式通過純代碼方式 總結Qt Creator中的快捷鍵如何使用文檔命名規范 簡介&#xff1a;這篇文章著重點并不在于創建hello world程序&#xff0c…

實時網絡流量監控與防御:Python實現DDoS攻擊檢測

1. 需求分析 DDoS攻擊通過海量請求耗盡服務器資源。本文使用 Python Scapy 實時監控流量&#xff0c;自動觸發IP封禁。 2. 核心代碼實現 2.1 依賴安裝 pip install scapy psutil2.2 流量監控腳本&#xff08;ddos_detector.py&#xff09; import time from scapy.all im…

電賽經驗分享——模塊篇

1、前言 打算在這一個專欄中&#xff0c;分享一些本科控制題電賽期間的經驗&#xff0c;和大家共同探討&#xff0c;也希望能幫助剛剛參加電賽的同學&#xff0c;了解一些基本的知識。一些見解和看法可能不同或有錯誤&#xff0c;歡迎批評指正。 在本文中&#xff0c;主要介紹筆…

【LLM】Open WebUI 使用指南:詳細圖文教程

Open WebUI 是一個開源的、可擴展且用戶友好的自托管 AI 平臺,專為生成式人工智能模型交互而設計。 Open WebUI 旨在為用戶提供一個簡單易用、功能強大且高度定制化的界面,使其能夠輕松與各種 AI 模型(如文本生成、圖像生成、語音識別等)進行交互。 一、安裝與初始化配置 擴…

HarmonyOS Next~HarmonyOS應用測試全流程解析:從一級類目上架到二級類目專項測試

HarmonyOS Next&#xff5e;HarmonyOS應用測試全流程解析&#xff1a;從一級類目上架到二級類目專項測試 引言&#xff1a;HarmonyOS生態下的質量保障挑戰 在萬物互聯的智能時代&#xff0c;HarmonyOS作為分布式操作系統&#xff0c;為開發者帶來了前所未有的創新空間&#x…

一種機載掃描雷達實時超分辨成像方法——論文閱讀

一種機載掃描雷達實時超分辨成像方法 1. 專利的研究目標與產業意義1.1 研究目標與實際問題1.2 產業意義2. 專利的創新方法:滑窗遞歸優化與實時更新2.1 核心模型與公式2.2 與傳統方法對比優勢3. 實驗設計與驗證3.1 仿真參數3.2 實驗結果4. 未來研究方向與挑戰4.1 學術挑戰4.2 技…

滾筒洗衣機拆解學習

本文圖片來自于B站視頻&#xff0c;鏈接在文末&#xff0c;不涉及任何公司及實驗室產品 對小米滾筒洗衣機進行拆解&#xff0c;并收集了用戶對這款產品的評價&#xff0c;認識了關鍵部件。下一步重點學習對各個電機的控制邏輯和供電系統。 整機拆解學習&#xff1a; 功能面板…

【金倉數據庫征文】金倉數據庫 KingbaseES 在電商平臺數據庫遷移與運維中深入復現剖析

【金倉數據庫征文】金倉數據庫 KingbaseES 在電商平臺數據庫遷移與運維中深入復現剖析 前言 在當今數字化商業蓬勃發展的時代&#xff0c;電商平臺的數據量呈爆發式增長&#xff0c;對數據庫性能、穩定性和擴展性提出了極高要求。本文章基于大型電商平臺原本采用 MySQL 數據庫&…

iPhone手機連接WiFi異常解決方法

iPhone手機連接WiFi異常解決方法 一、問題現象二、iPhone連不上可能的原因三、基礎排查與快速修復第一步:重啟大法第二步:忽略網絡,重新認證第三步:關閉“私有無線局域網地址”第四步:修改DNS服務器第五步:還原網絡設置四、路由器端排查及設置關閉MAC地址過濾或添加到白名…

Android NDK版本迭代與FFmpeg交叉編譯完全指南

在Android開發中&#xff0c;使用NDK(Native Development Kit)進行原生代碼開發是一項常見需求&#xff0c;特別是當我們需要集成FFmpeg這樣的多媒體處理庫時。本文將深入分析Android NDK的版本迭代分界線&#xff0c;詳細講解FFmpeg交叉編譯的注意事項&#xff0c;并提供完整的…

typecho中的Widget設計文檔

組成系統的最基本元素 什么是Widget Widget是組成Typecho的最基本元素&#xff0c;除了已經抽象出來的類庫外&#xff0c;其它幾乎所有的功能都會通過Widget來完成。在實踐中我們發現&#xff0c;在博客這種小型但很靈活的系統中實施一些大型框架的思想是不合適的&#xff0c…

Python序列Day3

序列 序列是一種數據存儲方式&#xff0c;用方括號標注&#xff0c;逗號分隔的一組值。在內存中&#xff0c;序列就是一塊用來存放多個值的連續的內存空間。 常見序列結構有&#xff1a;字符串、列表、元組、字典、集合 列表 用于存儲任意數目&#xff0c;任意類型的數據集…

私服與外掛:刑事法律風險的深度剖析

首席數據官高鵬律師團隊編著 在當今數字化時代&#xff0c;網絡游戲產業蓬勃發展&#xff0c;然而與之相伴的私服與外掛現象卻屢禁不止&#xff0c;且其背后隱藏著嚴重的刑事法律風險。作為一名律師&#xff0c;有必要在此對私服與外掛相關的刑事問題進行深入解讀&#xff0c;以…

Linux云計算訓練營筆記day04(Rocky Linux中的命令)

mv 移動(剪切) 源數據會消失 格式: mv 源文件 目標路徑 touch /opt/a.txt 創建文件 mv /opt/a.txt /root 移動文件&#xff0c;沒有改名 mkdir gongli 創建目錄 mv gongli /opt/ 移動目錄&#xff0c;沒有改名 mv /opt/gongli tedu 移動目錄&#xff0c;改名了 …

藍橋杯青少 圖形化編程——“星星”點燈

藍橋杯青少 圖形化編程——“星星”點燈 編程實現&#xff1a; 有10盞燈&#xff0c;從1到10按順序依次編號&#xff0c;初始時全部燈處于開啟狀態。有10個人也從1到10依次編號。第一個人&#xff08;1號&#xff09;將燈全部關閉&#xff0c;第二個人&#xff08;2號&#x…

conda配置好的pytorch在jupyter中如何配置

配置 其實不用再配置了 如下圖&#xff08;主要是激活pytorch環境&#xff0c;再jupyter notebook&#xff09; jupyter運行快捷鍵shiftenter 新建文件夾folder&#xff0c;新建notebook 使用 幫助文檔&#xff08;兩種方式&#xff09; ctrl/ 注釋

COLT_CMDB_aix_diskinfo.sh

#!/bin/ksh #IT_BEGIN #IT_TYPE3 #IT SYSTEM_AIX_AGENTDISKDISCOVER|discovery.diskInfo[disc] #原型指標 #IT_RULE SYSTEM_AIX_IP|ipAddress[{#DISKNAME}] #IT_RULE SYSTEM_AIX_AGENTDISKPATH|diskPath[{#DISKNAME}] #IT_RULE SYSTEM_AIX_DISKNAME|diskName[{#DISKNAME}] #IT_…