Synchronize的實現(xiàn)原理

1 Java對象頭信息

Java對象在JVM中的結(jié)構(gòu)如下:

java對象包括:

  • Mark Word(存儲對象的hashCode或者鎖信息)

  • Class Pointer(存儲對象所屬Class對象的指針)

  • length (只有數(shù)組對象才有這個字段)

  • Instance Data/Array Data (存儲實際數(shù)據(jù))

  • Padding (對齊緩存,填充對象被8整除的最小byte)

對象都在32/64位機器中每個部分分別是32/64位,Class Pointer在64位機器默認開啟指針壓縮,只占用32位。

2 對象加鎖過程

對象加鎖使用的是Mark Word字段,如下是32位的Mark Word

通過synchronize關(guān)鍵字給對象加鎖的過程如下:

  1. 無鎖:對象剛創(chuàng)建,未被加鎖;

  2. 偏向鎖:Mark Word存儲線程ID(默認開啟)

  3. 輕量級鎖(自旋鎖):Mark Word 存儲現(xiàn)在棧地址

  4. 重量級鎖:調(diào)用操作系統(tǒng)的metux(已經(jīng)不再JVM層面,發(fā)生了系統(tǒng)調(diào)用)

3 偏向鎖

JVM引入偏向鎖是為了在無多線程競爭的情況下盡量減少不必要的輕量級鎖執(zhí)行路徑,因為輕量級鎖的獲取及釋放依賴多次CAS原子指令,而偏向鎖只需要在置換ThreadID的時候依賴一次CAS原子指令(一旦出現(xiàn)多線程競爭的情況就必須撤銷偏向鎖)。

3.1偏向鎖獲取過程
  1. 訪問Mark Word中偏向鎖的標識是否設(shè)置成1,鎖標志位是否為01——確認為可偏向狀態(tài)。

  2. 如果為可偏向狀態(tài),則測試線程ID是否指向當前線程,如果是,進入步驟(5),否則進入步驟(3)。

  3. 如果線程ID并未指向當前線程,則通過CAS操作競爭鎖。如果競爭成功,則將Mark Word中線程ID設(shè)置為當前線程ID,然后執(zhí)行(5);如果競爭失敗,執(zhí)行(4)。

  4. 如果CAS獲取偏向鎖失敗,則表示有競爭。當?shù)竭_全局安全點(safepoint)時獲得偏向鎖的線程被掛起,偏向鎖升級為輕量級鎖,然后被阻塞在安全點的線程繼續(xù)往下執(zhí)行同步代碼。

  5. 執(zhí)行同步代碼。

3.2 偏向鎖的釋放

偏向鎖只有遇到其他線程嘗試競爭偏向鎖時,持有偏向鎖的線程才會釋放鎖,線程不會主動去釋放偏向鎖。偏向鎖的撤銷,需要等待全局安全點(在這個時間點上沒有字節(jié)碼正在執(zhí)行),它會首先暫停擁有偏向鎖的線程,判斷鎖對象是否處于被鎖定狀態(tài),撤銷偏向鎖后恢復到未鎖定(標志位為“01”)或輕量級鎖(標志位為“00”)的狀態(tài)。

4 輕量級鎖

輕量級鎖所適應的場景是線程交替執(zhí)行同步塊的情況,如果存在同一時間訪問同一鎖的情況,就會導致輕量級鎖膨脹為重量級鎖。

4.1 輕量級鎖的加鎖過程
  1. 代碼進入同步塊的時候,如果同步對象鎖狀態(tài)為無鎖狀態(tài),虛擬機將在當前線程的棧幀中建立一個名為鎖記錄(Lock Record)的空間;

  2. 拷貝對象頭中的Mark Word復制到鎖記錄中。

  3. 虛擬機將使用CAS操作嘗試將對象的Mark Word更新為指向Lock Record的指針,并將Lock record里的owner指針指向object mark word。如果更新成功,則執(zhí)行步驟(4),否則執(zhí)行步驟(5);

  4. 如果這個更新動作成功了,那么這個線程就擁有了該對象的鎖,并且對象Mark Word的鎖標志位設(shè)置為“00”,即表示此對象處于輕量級鎖定狀態(tài);

  5. 如果這個更新操作失敗了,虛擬機首先會檢查對象的Mark Word是否指向當前線程的棧幀,如果是就說明當前線程已經(jīng)擁有了這個對象的鎖,那就可以直接進入同步塊繼續(xù)執(zhí)行。否則說明多個線程競爭鎖,當前線程便嘗試使用自旋來獲取鎖,自旋失敗后,輕量級鎖就要膨脹為重量級鎖,鎖標志的狀態(tài)值變?yōu)椤?0”,Mark Word中存儲的就是指向重量級鎖(互斥量)的指針,后面等待鎖的線程也要進入阻塞狀態(tài)。

4.2 輕量級鎖的解鎖過程
  1. 通過CAS操作嘗試把線程中復制的 Mark Word 對象替換當前的 Mark Word。
  2. 如果替換成功,整個同步過程就完成了。
  3. 如果替換失敗,說明有其他線程嘗試過獲取該鎖(此時鎖已膨脹),那就要在釋放鎖的同時,喚醒被掛起的線程。

5 重量級鎖

synchronize的實現(xiàn)過程:

  1. java語言層級:synchronize關(guān)鍵字實現(xiàn)加鎖

  2. 字節(jié)碼層級:moniterenter,moniterexit字節(jié)碼實現(xiàn)加鎖

  3. JVM層級:偏向鎖->輕量級鎖->重量級鎖(已經(jīng)不在JVM,調(diào)用的操作系統(tǒng)的互斥量)

  4. 操作系統(tǒng)層級:申請操作系統(tǒng)的metux互斥鎖實現(xiàn)加鎖,會阻塞線程

  5. 硬件層級:通過 lock comxchg(CMS)實現(xiàn)獲取鎖資源,Lock前綴指令保障一致性

注意:lock前綴指令的功能:Synchronize, volatile,CMS都是使用這個實現(xiàn)

  1. 鎖總線,其它CPU對內(nèi)存的讀寫請求都會被阻塞,直到鎖釋放,不過實際后來的處理器都采用鎖緩存替代鎖總線,因為鎖總線的開銷比較大,鎖總線期間其他CPU沒法訪問內(nèi)存(原子性保障)
  2. lock后的寫操作會回寫已修改的數(shù)據(jù),同時讓其它CPU相關(guān)緩存行失效,從而重新從主存中加載最新的數(shù)據(jù)(可見性保障)
  3. 不是內(nèi)存屏障卻能完成類似內(nèi)存屏障的功能,阻止屏障兩遍的指令重排序(有序性保障)

當鎖膨脹成重量級鎖的時候,在JVM中當前鎖對象關(guān)聯(lián)的ObjectMonitor對象。

ObjectMonitor對象的數(shù)據(jù)結(jié)構(gòu)如下:

  • WaitSet:存放被wait()方法waiting的線程

  • EntryList:存放獲取不到鎖被blocking的線程

  • OwnerThread:指向當前獲取到鎖的線程

  • recursions:記錄鎖重入的次數(shù)

EntryList是一個后進先出的雙向鏈表,AQS(ReentrantLock)是一個先進先出的雙向鏈表。

ObjectMoniter的流程:

  1. 拿到鎖的線程修改ObjectMoniter的OwnerThread指向自己,recursions加一

  2. 未拿到鎖的線程通過操作系統(tǒng)的阻塞,添加到EntryList隊列中

  3. 當線程調(diào)用wait()方法時,線程進入Waitset集合,并釋放鎖

  4. 當調(diào)用notify()方法時,從Waitset集合取出,放入EntryList隊尾

注意:

Synchronize只有一個WaitSet,AQS可以創(chuàng)建多個Condition隊列(功能和Waitset類似)。

6 三種鎖的比較

synchronize的實現(xiàn)原理_技術(shù)流水-CSDN博客_synchronize

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,461評論 6 532
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,538評論 3 417
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,423評論 0 375
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,991評論 1 312
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,761評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,207評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,268評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,419評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,959評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 40,782評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,983評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,528評論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 44,222評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,653評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,901評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,678評論 3 392
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,978評論 2 374

推薦閱讀更多精彩內(nèi)容