成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久

您的位置:首頁技術文章
文章詳情頁

Java同步關鍵字synchronize底層實現原理解析

瀏覽:61日期:2023-12-05 08:51:52
目錄1 字節碼層實現1.1 InterpreterRuntime::monitorenter1.1.1 函數參數 JavaThread *thread1.1.2 函數體 2 偏向鎖 2.1 偏向鎖的意義2.2 偏向鎖的獲取2.2.1 markOop mark = obj->mark()2.2.2 判斷mark是否為可偏向狀態2.2.3 判斷mark中JavaThread的狀態2.2.4 通過CAS原子指令2.2.5 如果執行CAS失敗2.3 偏向鎖的撤銷2.4 輕量級鎖2.4.1 引入輕量級鎖的目的2.4.2 輕量級鎖的獲取總結1 字節碼層實現

javap 生成的字節碼中包含如下指令:

monitorenter monitorexit

synchronized基此實現了簡單直接的鎖的獲取和釋放。

當JVM的解釋器執行monitorenter時會進入到InterpreterRuntime.cpp的

1.1 InterpreterRuntime::monitorenter1.1.1 函數參數 JavaThread *thread

封裝 Java線程 幀狀態的與機器/操作系統相關的部分的對象,這里傳參代表程序中的當前線程BasicObjectLock *elem

Java同步關鍵字synchronize底層實現原理解析

BasicLock 類型的 _lock 對象主要用來保存 _obj 對象的對象頭數據:

Java同步關鍵字synchronize底層實現原理解析

1.1.2 函數體

Java同步關鍵字synchronize底層實現原理解析

UseBiasedLocking 標識JVM是否開啟偏向鎖功能

如果開啟則執行fast_enter邏輯 否則執行slow_enter 2 偏向鎖 2.1 偏向鎖的意義

無多線程競爭時,盡量減少不必要的輕量級鎖執行路徑。

輕量級鎖的獲取及釋放依賴多次的CAS操作,而偏向鎖只依賴一次CAS置換ThreadID。

當存在高度的鎖競爭和低數據競爭時,RTM 鎖最有用。高鎖爭用情況下,鎖通常會膨脹,而偏向鎖不適于這種情況。RTM 鎖代碼要求關閉偏向鎖。

注意:我們不能在 get_processor_features() 中關閉 UseBiasedLocking,因為它被 Thread::allocate() 使用,它在 VM_Version::initialize() 之前調用。

if (UseRTMLocking && UseBiasedLocking) { if (FLAG_IS_DEFAULT(UseBiasedLocking)) { FLAG_SET_DEFAULT(UseBiasedLocking, false); } else { warning('Biased locking is not supported with RTM locking; ignoring UseBiasedLocking flag.' ); UseBiasedLocking = false; }}

一旦出現多個線程競爭時必須撤銷偏向鎖,所以:

撤銷偏向鎖消耗的性能必須 < 之前節省下來的CAS原子操作的性能消耗

不然得不償失!

JDK 6中默認開啟偏向鎖,可以通過-XX:-UseBiasedLocking禁用偏向鎖。

偏向鎖的入口位于synchronizer.cpp文件的ObjectSynchronizer::fast_enter函數

Java同步關鍵字synchronize底層實現原理解析

2.2 偏向鎖的獲取

由BiasedLocking::revoke_and_rebias方法實現

Java同步關鍵字synchronize底層實現原理解析

2.2.1 markOop mark = obj->mark()

獲取對象的markOop數據mark,即對象頭的Mark Word

Java同步關鍵字synchronize底層實現原理解析

2.2.2 判斷mark是否為可偏向狀態 mark的偏向鎖標志位為 1 鎖標志位為 012.2.3 判斷mark中JavaThread的狀態

如果為空,則進入步驟(4);如果指向當前線程,則執行同步代碼塊;如果指向其它線程,進入步驟(5);

2.2.4 通過CAS原子指令

設置mark中JavaThread為當前線程ID,如果執行CAS成功,則執行同步代碼塊,否則進入步驟(5);

2.2.5 如果執行CAS失敗

表示當前存在多個線程競爭鎖,當達到全局安全點(safepoint),獲得偏向鎖的線程被掛起,撤銷偏向鎖,并升級為輕量級,升級完成后被阻塞在安全點的線程繼續執行同步代碼塊;

2.3 偏向鎖的撤銷

只有當其它線程嘗試競爭偏向鎖時,持有偏向鎖的線程才會釋放鎖,偏向鎖的撤銷由BiasedLocking::revoke_at_safepoint方法實現:

Java同步關鍵字synchronize底層實現原理解析

1、偏向鎖的撤銷動作必須等待全局安全點;2、暫停擁有偏向鎖的線程,判斷鎖對象是否處于被鎖定狀態;3、撤銷偏向鎖,恢復到無鎖(標志位為 01)或輕量級鎖(標志位為 00)的狀態;

偏向鎖在Java 1.6之后是默認啟用的,但在應用程序啟動幾秒鐘之后才激活,可以使用-XX:BiasedLockingStartupDelay=0參數關閉延遲,如果確定應用程序中所有鎖通常情況下處于競爭狀態,可以通過XX:-UseBiasedLocking=false參數關閉偏向鎖。

2.4 輕量級鎖2.4.1 引入輕量級鎖的目的

在多線程交替執行同步塊的情況下,盡量避免重量級鎖引起的性能消耗,但是如果多個線程在同一時刻進入臨界區,會導致輕量級鎖膨脹升級重量級鎖,所以輕量級鎖的出現并非是要替代重量級鎖

2.4.2 輕量級鎖的獲取

當關閉偏向鎖功能,或多個線程競爭偏向鎖導致偏向鎖升級為輕量級鎖,會嘗試獲取輕量級鎖,其入口位于ObjectSynchronizer::slow_enter

Java同步關鍵字synchronize底層實現原理解析

1、markOop mark = obj->mark()方法獲取對象的markOop數據mark;2、mark->is_neutral()方法判斷mark是否為無鎖狀態:mark的偏向鎖標志位為 0,鎖標志位為 01;3、如果mark處于無鎖狀態,則進入步驟(4),否則執行步驟(6);4、把mark保存到BasicLock對象的_displaced_header字段;5、通過CAS嘗試將Mark Word更新為指向BasicLock對象的指針,如果更新成功,表示競爭到鎖,則執行同步代碼,否則執行步驟(6);6、如果當前mark處于加鎖狀態,且mark中的ptr指針指向當前線程的棧幀,則執行同步代碼,否則說明有多個線程競爭輕量級鎖,輕量級鎖需要膨脹升級為重量級鎖;

假設線程A和B同時執行到臨界區if (mark->is_neutral()):1、線程AB都把Mark Word復制到各自的_displaced_header字段,該數據保存在線程的棧幀上,是線程私有的;2、Atomic::cmpxchg_ptr原子操作保證只有一個線程可以把指向棧幀的指針復制到Mark Word,假設此時線程A執行成功,并返回繼續執行同步代碼塊;3、線程B執行失敗,退出臨界區,通過ObjectSynchronizer::inflate方法開始膨脹鎖;

輕量級鎖的釋放

輕量級鎖的釋放通過ObjectSynchronizer::fast_exit完成。

Java同步關鍵字synchronize底層實現原理解析

1、確保處于偏向鎖狀態時不會執行這段邏輯;2、取出在獲取輕量級鎖時保存在BasicLock對象的mark數據dhw;3、通過CAS嘗試把dhw替換到當前的Mark Word,如果CAS成功,說明成功的釋放了鎖,否則執行步驟(4);4、如果CAS失敗,說明有其它線程在嘗試獲取該鎖,這時需要將該鎖升級為重量級鎖,并釋放;

重量級鎖

重量級鎖通過對象內部的監視器(monitor)實現,其中monitor的本質是依賴于底層操作系統的Mutex Lock實現,操作系統實現線程之間的切換需要從用戶態到內核態的切換,切換成本非常高。

鎖膨脹過程

鎖的膨脹過程通過ObjectSynchronizer::inflate函數實現

Java同步關鍵字synchronize底層實現原理解析

膨脹過程的實現比較復雜,截圖中只是一小部分邏輯,完整的方法可以查看synchronized.cpp,大概實現過程如下:1、整個膨脹過程在自旋下完成;2、mark->has_monitor()方法判斷當前是否為重量級鎖,即Mark Word的鎖標識位為 10,如果當前狀態為重量級鎖,執行步驟(3),否則執行步驟(4);3、mark->monitor()方法獲取指向ObjectMonitor的指針,并返回,說明膨脹過程已經完成;4、如果當前鎖處于膨脹中,說明該鎖正在被其它線程執行膨脹操作,則當前線程就進行自旋等待鎖膨脹完成,這里需要注意一點,雖然是自旋操作,但不會一直占用cpu資源,每隔一段時間會通過os::NakedYield方法放棄cpu資源,或通過park方法掛起;如果其他線程完成鎖的膨脹操作,則退出自旋并返回;5、如果當前是輕量級鎖狀態,即鎖標識位為 00,膨脹過程如下:

Java同步關鍵字synchronize底層實現原理解析

1、通過omAlloc方法,獲取一個可用的ObjectMonitor monitor,并重置monitor數據;2、通過CAS嘗試將Mark Word設置為markOopDesc:INFLATING,標識當前鎖正在膨脹中,如果CAS失敗,說明同一時刻其它線程已經將Mark Word設置為markOopDesc:INFLATING,當前線程進行自旋等待膨脹完成;3、如果CAS成功,設置monitor的各個字段:_header、_owner和_object等,并返回;

monitor競爭

當鎖膨脹完成并返回對應的monitor時,并不表示該線程競爭到了鎖,真正的鎖競爭發生在ObjectMonitor::enter方法中。

Java同步關鍵字synchronize底層實現原理解析

1、通過CAS嘗試把monitor的_owner字段設置為當前線程;2、如果設置之前的_owner指向當前線程,說明當前線程再次進入monitor,即重入鎖,執行_recursions ++ ,記錄重入的次數;3、如果之前的_owner指向的地址在當前線程中,這種描述有點拗口,換一種說法:之前_owner指向的BasicLock在當前線程棧上,說明當前線程是第一次進入該monitor,設置_recursions為1,_owner為當前線程,該線程成功獲得鎖并返回;4、如果獲取鎖失敗,則等待鎖的釋放;

monitor等待

monitor競爭失敗的線程,通過自旋執行ObjectMonitor::EnterI方法等待鎖的釋放,EnterI方法的部分邏輯實現如下:

Java同步關鍵字synchronize底層實現原理解析

1、當前線程被封裝成ObjectWaiter對象node,狀態設置成ObjectWaiter::TS_CXQ;2、在for循環中,通過CAS把node節點push到_cxq列表中,同一時刻可能有多個線程把自己的node節點push到_cxq列表中;3、node節點push到_cxq列表之后,通過自旋嘗試獲取鎖,如果還是沒有獲取到鎖,則通過park將當前線程掛起,等待被喚醒,實現如下:

Java同步關鍵字synchronize底層實現原理解析

4、當該線程被喚醒時,會從掛起的點繼續執行,通過ObjectMonitor::TryLock嘗試獲取鎖,TryLock方法實現如下:

Java同步關鍵字synchronize底層實現原理解析

其本質就是通過CAS設置monitor的_owner字段為當前線程,如果CAS成功,則表示該線程獲取了鎖,跳出自旋操作,執行同步代碼,否則繼續被掛起;

monitor釋放

當某個持有鎖的線程執行完同步代碼塊時,會進行鎖的釋放,給其它線程機會執行同步代碼,在HotSpot中,通過退出monitor的方式實現鎖的釋放,并通知被阻塞的線程,具體實現位于ObjectMonitor::exit方法中。

Java同步關鍵字synchronize底層實現原理解析

1、如果是重量級鎖的釋放,monitor中的_owner指向當前線程,即THREAD == _owner;2、根據不同的策略(由QMode指定),從cxq或EntryList中獲取頭節點,通過ObjectMonitor::ExitEpilog方法喚醒該節點封裝的線程,喚醒操作最終由unpark完成,實現如下:

void ObjectMonitor::ExitEpilog(Thread * Self, ObjectWaiter * Wakee) { assert(_owner == Self, 'invariant'); // Exit protocol: // 1. ST _succ = wakee // 2. membar #loadstore|#storestore; // 2. ST _owner = NULL // 3. unpark(wakee) _succ = Wakee->_thread; ParkEvent * Trigger = Wakee->_event; // Hygiene -- once we’ve set _owner = NULL we can’t safely dereference Wakee again. // The thread associated with Wakee may have grabbed the lock and 'Wakee' may be // out-of-scope (non-extant). Wakee = NULL; // Drop the lock OrderAccess::release_store(&_owner, (void*)NULL); OrderAccess::fence(); // ST _owner vs LD in unpark() DTRACE_MONITOR_PROBE(contended__exit, this, object(), Self); Trigger->unpark(); // Maintain stats and report events to JVMTI OM_PERFDATA_OP(Parks, inc());}

3、被喚醒的線程,繼續執行monitor的競爭;

總結

到此這篇關于Java同步關鍵字synchronize底層實現原理解析的文章就介紹到這了,更多相關java synchronize底層實現原理內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Java
相關文章:
成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久
免费国产一区二区| 亚洲国产一区二区三区高清| 激情综合色播五月| 影音欧美亚洲| 欧美一级生活片| 亚洲精品免费在线观看| 日韩精品1区2区3区| 欧美日韩调教| 69堂成人精品免费视频| 亚洲欧美另类在线| 成人综合激情网| 午夜精品久久久久久久99樱桃| 日韩一区欧美二区| 久久精品一二三区| 国产精品午夜在线| 国产精品视频看| 亚洲欧美久久久| 日韩美一区二区三区| 日韩一区二区麻豆国产| 亚洲二区视频在线| 欧美日本一区二区三区| 久久最新视频| 久久免费的精品国产v∧| 免费的国产精品| 国产婷婷精品| 粉嫩绯色av一区二区在线观看| 欧美aⅴ99久久黑人专区| 91精品国产综合久久精品图片| 中文字幕中文字幕一区| 国产不卡免费视频| 色美美综合视频| 亚洲理论在线观看| 91丝袜国产在线播放| 91麻豆精品国产91久久久资源速度| 亚洲永久精品大片| 欧美体内she精视频在线观看| 欧美久久一二区| 亚洲成av人片在线观看| 亚洲黄网站黄| 亚洲国产精品传媒在线观看| 国产激情视频一区二区在线观看| 色婷婷综合五月| 亚洲国产精品久久艾草纯爱| 狠色狠色综合久久| 国产人成一区二区三区影院| 国产a级毛片一区| 欧美在线啊v一区| 日本不卡中文字幕| 亚洲免费久久| 中文字幕欧美一区| 成人黄色在线网站| 欧美一区二区在线看| 日韩高清不卡在线| 亚洲专区免费| 一区二区三区在线免费视频| 91蜜桃传媒精品久久久一区二区| 欧美男人的天堂一二区| 日韩国产高清影视| 亚洲麻豆av| 亚洲女人****多毛耸耸8| 99精品久久| 亚洲最快最全在线视频| 99日韩精品| 亚洲三级久久久| 欧美不卡视频| 欧美mv日韩mv国产网站app| 日韩电影在线免费观看| 老司机午夜精品视频| 日韩精品五月天| 日本高清不卡视频| 亚洲一区二区三区视频在线| 亚洲理论在线| 亚洲免费观看高清在线观看| 狠狠88综合久久久久综合网| 国产精品视频观看| 欧美视频不卡| 国产精品久久久久久妇女6080| 欧美理论在线| 国产精品免费视频观看| 欧美日韩在线精品| 国产精品国产a级| 在线成人www免费观看视频| 欧美体内she精高潮| 久久99精品网久久| 欧美一区二区精品在线| 成人a区在线观看| 国产无一区二区| 欧美日韩精品免费观看| 亚洲色图色小说| 亚洲女人av| 捆绑调教一区二区三区| 91精品国产入口| 成人av资源站| 国产精品看片你懂得| 麻豆久久婷婷| 在线观看国产精品网站| 99视频在线精品| 日韩精品高清不卡| 国产欧美一区二区精品婷婷| 在线观看免费视频综合| 欧美日韩国产欧| 精品一区二区三区蜜桃| 国产精品久久久久久亚洲伦 | 亚洲欧洲精品一区二区精品久久久 | 91视频在线看| 亚洲成在人线在线播放| 欧美大片一区二区| 久久精品伊人| 国内一区二区在线视频观看| 国产另类ts人妖一区二区| 亚洲午夜影视影院在线观看| 精品88久久久久88久久久| 精品国产免费一区二区三区香蕉 | 黑人巨大精品欧美一区二区小视频| 欧美日本视频在线| 色综合天天做天天爱| 免费欧美在线视频| 国产免费成人在线视频| 欧美电影一区二区三区| 亚洲视频1区| av亚洲精华国产精华精华| 日韩av电影免费观看高清完整版 | 国产白丝网站精品污在线入口| 一区二区三区四区亚洲| 久久综合资源网| 在线精品视频免费播放| 99re国产精品| 亚洲欧美日韩专区| 欧美日产一区二区三区在线观看| 国产激情视频一区二区三区欧美 | 国产欧美一区二区色老头| 久久久精品免费免费| 播五月开心婷婷综合| 精品国产乱码久久久久久1区2区| 国产精品v欧美精品v日本精品动漫| 久久er精品视频| 亚洲色图都市小说| 日韩精品一区国产麻豆| 激情久久综合| 成人福利视频网站| 免费日本视频一区| 亚洲福中文字幕伊人影院| 欧美激情一区二区三区在线| 日韩一级大片在线观看| 在线免费观看日韩欧美| 国产亚洲在线| 韩日成人在线| 色综合色狠狠天天综合色| 成人午夜精品一区二区三区| 狠狠色丁香久久婷婷综合_中 | 97se亚洲国产综合在线| 国产精品一区二区久激情瑜伽| 亚洲午夜精品一区二区三区他趣| 国产欧美一区二区精品久导航 | 欧美日韩精品是欧美日韩精品| 久久九九免费| 国产精品久久久久久久免费软件| 欧美日韩一区二区三区在线观看免 | 日本麻豆一区二区三区视频| 亚洲va国产va欧美va观看| 一区二区三区资源| 日韩久久一区二区| 亚洲国产成人午夜在线一区| 久久久青草青青国产亚洲免观| 日韩视频一区在线观看| 欧美一区二区三区小说| 欧美一区二区三区免费在线看| 欧美群妇大交群的观看方式 | 国产精品护士白丝一区av| 久久久久久久久久美女| 欧美变态凌虐bdsm| 日韩一级完整毛片| 日韩一区二区在线看| 51午夜精品国产| 欧美一级片免费看| 91麻豆精品国产91久久久久久久久| 欧美性欧美巨大黑白大战| 欧美性生活影院| 欧美三区免费完整视频在线观看| 欧洲精品在线观看| 欧美日本精品一区二区三区| 欧美狂野另类xxxxoooo| 色婷婷久久久综合中文字幕| 欧美中文字幕一区| 欧美日韩情趣电影| 日韩视频一区二区三区在线播放| 精品国产区一区| 国产欧美一区二区精品性色 | 久久久综合网| 色婷婷综合视频在线观看| 欧洲精品一区二区| 777a∨成人精品桃花网| 日韩欧美国产电影| 精品少妇一区二区三区在线播放 | 精品不卡在线| 日韩一级在线| 香蕉成人久久| 在线看国产日韩| 在线播放中文字幕一区| 欧美电影精品一区二区|