詳解Java對象結(jié)構(gòu)與對象鎖的升級
1. Java對象結(jié)構(gòu)
Java對象結(jié)構(gòu)包括三部分:對象頭、對象體和填充字節(jié),如圖所示:
對象頭又包括三個字段:
- 第一個字段叫作
Mark Word
(標(biāo)記字),用于存儲自身運(yùn)行時的數(shù)據(jù),例如GC標(biāo)志位、哈希碼、鎖狀態(tài)等信息。 - 第二個字段叫作
Class Pointer
(類對象指針),用于存放方法區(qū)Class對象的地址,虛擬機(jī)通過這個指針來確定這個對象是哪個類的實例。 - 第三個字段叫作
Array Length
(數(shù)組長度)。如果對象是一個Java數(shù)組,那么此字段必須有,用于記錄數(shù)組長度的數(shù)據(jù);如果對象不是一個Java數(shù)組,那么此字段不存在,所以這是一個可選字段。
在32位JVM虛擬機(jī)中,Mark Word和Class Pointer這兩部分都是32位的;在64位JVM虛擬機(jī)中,Mark Word和Class Pointer這兩部分都是64位的。而我們需要重點(diǎn)理解的時mark word , 因為它跟synchronized的底層原理有關(guān)。
2. Mark Word的結(jié)構(gòu)信息
Java內(nèi)置鎖的狀態(tài)總共有4種,級別由低到高依次為:無鎖、偏向鎖、輕量級鎖和重量級鎖。其實在JDK 1.6之前,Java內(nèi)置鎖還是一個重量級鎖,是一個效率比較低下的鎖,在JDK 1.6之后,JVM為了提高鎖的獲取與釋放效率,對synchronized的實現(xiàn)進(jìn)行了優(yōu)化,引入了偏向鎖和輕量級鎖,從此以后Java內(nèi)置鎖的狀態(tài)就有了4種,并且4種狀態(tài)會隨著競爭的情況逐漸升級,而且是不可逆的過程,即不可降級,也就是說只能進(jìn)行鎖升級。
1、不同鎖狀態(tài)下的Mark Word字段結(jié)構(gòu):
Mark Word字段的結(jié)構(gòu)與Java內(nèi)置鎖的狀態(tài)強(qiáng)相關(guān)。為了讓Mark Word字段存儲更多的信息,JVM將Mark Word最低兩個位設(shè)置為Java內(nèi)置鎖狀態(tài)位,不同鎖狀態(tài)下的32位Mark Word結(jié)構(gòu)如表所示:
64位的Mark Word與32位的Mark Word結(jié)構(gòu)相似,結(jié)構(gòu)如表所示:
2、64位Mark Word的介紹:
由于目前主流的JVM都是64位,因此我們使用64位的Mark Word。接下來對64位的Mark Word中各部分的內(nèi)容進(jìn)行具體介紹。
(1) lock:鎖狀態(tài)標(biāo)記位,占兩個二進(jìn)制位,由于希望用盡可能少的二進(jìn)制位表示盡可能多的信息,因此設(shè)置了lock標(biāo)記。該標(biāo)記的值不同,整個Mark Word表示的含義就不同。
(2) biased_lock:對象是否啟用偏向鎖標(biāo)記,只占1個二進(jìn)制位。為1時表示對象啟用偏向鎖,為0時表示對象沒有偏向鎖。lock和biased_lock兩個標(biāo)記位組合在一起共同表示Object實例處于什么樣的鎖狀態(tài)。二者組合的含義具體如表2-3所示。
lock和biased_lock兩個標(biāo)記位組合在一起共同表示Object實例處于什么樣的鎖狀態(tài)。二者組合的含義具體如表所示:
(3) ptr_to_lock_record:占62位,在輕量級鎖的狀態(tài)下指向棧幀中鎖記錄的指針。
(4) ptr_to_heavyweight_monitor:占62位,在重量級鎖的狀態(tài)下指向?qū)ο蟊O(jiān)視器的指針。
3. 無鎖、偏向鎖、輕量級鎖和重量級鎖
在JDK 1.6版本之前,所有的Java內(nèi)置鎖都是重量級鎖。重量級鎖會造成CPU在用戶態(tài)和核心態(tài)之間頻繁切換,所以代價高、效率低。JDK 1.6版本為了減少獲得鎖和釋放鎖所帶來的性能消耗,引入了偏向鎖和輕量級鎖的實現(xiàn)。所以,在JDK 1.6版本中內(nèi)置鎖一共有4種狀態(tài):無鎖狀態(tài)、偏向鎖狀態(tài)、輕量級鎖狀態(tài)和重量級鎖狀態(tài),這些狀態(tài)隨著競爭情況逐漸升級。內(nèi)置鎖可以升級但不能降級,意味著偏向鎖升級成輕量級鎖后不能再降級成偏向鎖。這種能升級卻不能降級的策略,其目的是提高獲得鎖和釋放鎖的效率。
(1) 無鎖狀態(tài) :
Java對象剛創(chuàng)建時還沒有任何線程來競爭,說明該對象處于無鎖狀態(tài)(無線程競爭它),這時偏向鎖標(biāo)識位是0,鎖狀態(tài)是01;
(2) 偏向鎖狀態(tài):
偏向鎖是指一段同步代碼一直被同一個線程所訪問,那么該線程會自動獲取鎖,降低獲取鎖的代價。如果內(nèi)置鎖處于偏向狀態(tài),當(dāng)有一個線程來競爭鎖時,先用偏向鎖,表示內(nèi)置鎖偏愛這個線程,這個線程要執(zhí)行該鎖關(guān)聯(lián)的同步代碼時,不需要再做任何檢查和切換。偏向鎖在競爭不激烈的情況下效率非常高。
偏向鎖狀態(tài)的Mark Word會記錄內(nèi)置鎖自己偏愛的線程ID,內(nèi)置鎖會將該線程當(dāng)作自己的熟人,這時偏向鎖標(biāo)識位是1,鎖狀態(tài)是01;
(3) 輕量級鎖狀態(tài):
當(dāng)有兩個線程開始競爭這個鎖對象時,情況就發(fā)生變化了,不再是偏向(獨(dú)占)鎖了,鎖會升級為輕量級鎖,兩個線程公平競爭,哪個線程先占有鎖對象,鎖對象的Mark Word就指向哪個線程的棧幀中的鎖記錄。這時偏向鎖標(biāo)識位是0,鎖狀態(tài)是00;
當(dāng)鎖處于偏向鎖,又被另一個線程企圖搶占時,偏向鎖就會升級為輕量級鎖。企圖搶占的線程會通過自旋的形式嘗試獲取鎖,不會阻塞搶鎖線程,以便提高性能。
自旋原理非常簡單,如果持有鎖的線程能在很短時間內(nèi)釋放鎖資源,那么那些等待競爭鎖的線程就不需要進(jìn)行內(nèi)核態(tài)和用戶態(tài)之間的切換來進(jìn)入阻塞掛起狀態(tài),它們只需要等一等(自旋),等持有鎖的線程釋放鎖后即可立即獲取鎖,這樣就避免了用戶線程和內(nèi)核切換的消耗。
但是,線程自旋是需要消耗CPU的,如果一直獲取不到鎖,那么線程也不能一直占用CPU自旋做無用功,所以需要設(shè)定一個自旋等待的最大時間。JVM對于自旋周期的選擇,JDK 1.6之后引入了適應(yīng)性自旋鎖,適應(yīng)性自旋鎖意味著自旋的時間不是固定的,而是由前一次在同一個鎖上的自旋時間以及鎖的擁有者的狀態(tài)來決定的。線程如果自旋成功了,下次自旋的次數(shù)就會更多,如果自旋失敗了,自旋的次數(shù)就會減少。
如果持有鎖的線程執(zhí)行的時間超過自旋等待的最大時間仍沒有釋放鎖,就會導(dǎo)致其他爭用鎖的線程在最大等待時間內(nèi)還是獲取不到鎖,自旋不會一直持續(xù)下去,這時爭用線程會停止自旋進(jìn)入阻塞狀態(tài),該鎖膨脹為重量級鎖。
(4) 重量級鎖狀態(tài):
重量級鎖會讓其他申請的線程之間進(jìn)入阻塞,性能降低。重量級鎖也叫同步鎖,這個鎖對象MarkWord再次發(fā)生變化,會指向一個監(jiān)視器對象,該監(jiān)視器對象用集合的形式來登記和管理排隊的線程。這時偏向鎖標(biāo)識位是0,鎖狀態(tài)是10;
總結(jié)
本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
Java使用正則表達(dá)式去除小數(shù)點(diǎn)后面多余的0功能示例
這篇文章主要介紹了Java使用正則表達(dá)式去除小數(shù)點(diǎn)后面多余的0功能,結(jié)合具體實例形式分析了java字符串正則替換相關(guān)操作技巧,需要的朋友可以參考下2017-06-06詳解Java數(shù)組擴(kuò)容縮容與拷貝的實現(xiàn)和原理
這篇文章主要帶大家學(xué)習(xí)數(shù)組的擴(kuò)容、縮容及拷貝,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-05-05springboot如何使用logback-spring配置日志格式,并分環(huán)境配置
這篇文章主要介紹了springboot如何使用logback-spring配置日志格式,并分環(huán)境配置的操作,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-07-07SpringBoot可視化接口開發(fā)工具magic-api的簡單使用教程
作為Java后端開發(fā),平時開發(fā)API接口的時候經(jīng)常需要定義Controller、Service、Dao、Mapper、XML、VO等Java對象。有沒有什么辦法可以讓我們不寫這些代碼,直接操作數(shù)據(jù)庫生成API接口呢?今天給大家推薦一款工具magic-api,來幫我們實現(xiàn)這個小目標(biāo)!2021-06-06Java中的interrupt、interrupted和isInterrupted方法區(qū)別詳解
這篇文章主要介紹了Java中的interrupt、interrupted和isInterrupted方法區(qū)別詳解,interrupt用于中斷線程,調(diào)用該方法的線程的狀態(tài)將會被設(shè)置為中斷狀態(tài),線程中斷僅僅是設(shè)置線程的中斷狀態(tài)位,并不會停止線程,需要用戶自己去監(jiān)視線程的狀態(tài)并作出處理,需要的朋友可以參考下2023-12-12Mybatis返回int或者Integer類型報錯的解決辦法
這篇文章主要介紹了Mybatis返回int或者Integer類型報錯的解決辦法,非常不錯,具有參考借鑒價值,需要的朋友可以參考下2016-12-12Spring容器-BeanFactory和ApplicationContext使用詳解
這篇文章主要為大家介紹了Spring容器-BeanFactory和ApplicationContext的使用示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-04-04