深入詳解Java中synchronized鎖升級的套路
synchronized鎖是啥?鎖其實(shí)就是一個(gè)對象,隨便哪一個(gè)都可以,Java中所有的對象都是鎖,換句話說,Java中所有對象都可以成為鎖。
這次我們主要聊的是synchronized鎖升級的套路
synchronized會經(jīng)歷四個(gè)階段:無鎖狀態(tài)、偏向鎖、輕量級鎖、重量級鎖依次從耗費(fèi)資源最少,性能最高,到耗費(fèi)資源多,性能最差。
鎖原理
先看看這些狀態(tài)的鎖為什么稱之為鎖,他們的互斥原理是啥。
偏向鎖
當(dāng)一個(gè)線程到達(dá)同步代碼塊,嘗試獲取鎖對象的時(shí)候,會查看對象頭中的MarkWord里的線程ID,如果這里沒有ID則將自己的保存進(jìn)去,拿到鎖。若是有,則查看是否是當(dāng)前線程,如果不是,就CAS嘗試改,如果是,就已經(jīng)拿到了鎖資源。
這里詳細(xì)說說CAS嘗試修改的邏輯:它會檢查持有偏向鎖的線程狀態(tài)。首先遍歷當(dāng)前JVM的所有存活的線程,如果能找到偏向的線程,則說明偏向的線程還存活,此時(shí)會檢查線程是否在執(zhí)行同步代碼塊中的代碼,如果是,則升級為輕量級鎖,去繼續(xù)進(jìn)行CAS競爭鎖。所以加了偏向鎖之后,同時(shí)只有一個(gè)線程可以拿到鎖執(zhí)行同步代碼塊中的代碼。
輕量級鎖
查看對象頭中的MarkWord里的Lock Record指針指向的是否是當(dāng)前線程的虛擬機(jī)棧,如果是,拿鎖執(zhí)行業(yè)務(wù),如果不是則進(jìn)行CAS,嘗試修改,若是修改幾次都沒有成功,再升級到重量級鎖。
重量級鎖
查看對象頭中的MarkWord里的指向的ObjectMonitor,查看owner是否是當(dāng)前線程,如果不是,扔到ObjectMonitor里的EntryList中排隊(duì),并掛起線程,等待被喚醒。
鎖升級
無鎖
一般情況下,新new出來的一個(gè)對象,暫時(shí)就是無鎖狀態(tài)。因?yàn)槠蜴i默認(rèn)是有延遲的,在啟動JVM的前4s中,不存在偏向鎖,但是如果關(guān)閉了偏向鎖延遲的設(shè)置,new出來的對象,就會添加一個(gè)匿名偏向鎖。也就是說這個(gè)對象想找一個(gè)線程去增加偏向鎖,但是沒有找到,稱之為匿名偏向。存儲的線程ID為一堆0000,也沒有任何地址信息。
我們可以通過以下配置關(guān)閉偏向鎖延遲。
//關(guān)閉偏向鎖延遲的指令 -XX:BiasedLockingStartuoDelay=0
偏向鎖
當(dāng)某一個(gè)線程來獲取這個(gè)鎖資源時(shí),此時(shí)會成功獲取到,就會變?yōu)槠蜴i,偏向鎖存儲線程的ID。
當(dāng)偏向鎖升級時(shí),會觸發(fā)偏向鎖撤銷,偏向鎖撤銷需要等到一個(gè)安全點(diǎn),比如GC的時(shí)候,偏向鎖撤銷的成本太高,所以默認(rèn)開始時(shí),會做偏向鎖延遲。若是直接有多個(gè)線程競爭,會跳過偏向鎖,直接變?yōu)檩p量級鎖。
細(xì)說一下偏向鎖撤銷的過程,成本為啥高呢?當(dāng)一個(gè)線程拿到偏向鎖之后,會把鎖的對象頭的Mark Work中的線程id指向自己,當(dāng)又有一個(gè)線程來了進(jìn)行爭搶導(dǎo)致鎖升級的的時(shí)候,會暫停之前拿到偏向鎖的線程,然后清空Mark Work中的線程id,增加一個(gè)輕量級鎖,然后再恢復(fù)暫停的線程繼續(xù)執(zhí)行。這也是為什么等到安全點(diǎn)再執(zhí)行鎖升級的原因,因?yàn)橐獣和>€程。
常見的安全點(diǎn):
- 執(zhí)行GC的時(shí)候
- 方法返回之前
- 調(diào)用某個(gè)方法之后
- 拋出異常的位置
- 一個(gè)循環(huán)的末尾
輕量級鎖
當(dāng)在出現(xiàn)了多個(gè)線程的競爭,就會升級為輕量級鎖,輕量級鎖的效果就是基于CAS嘗試獲取鎖資源,這里會用到自適應(yīng)自旋鎖,根據(jù)上次CAS成功與否,耗費(fèi)的時(shí)間,決定這次自旋多少次。
輕量級鎖適用于競爭不是很激烈的場景,一個(gè)線程拿到鎖,執(zhí)行同步代碼塊,很快就處理完了。再來一個(gè)線程嘗試一兩次也拿到了鎖,再去執(zhí)行,不會讓一個(gè)線程等待很久。
重量級鎖
如果到了重量級鎖,那就沒啥說的了,如果有線程持有鎖,其他想拿鎖的就掛起,等待鎖釋放后被依次喚醒。
鎖粗化&鎖消除
鎖粗化/鎖膨脹
鎖膨脹是編譯Java文件的時(shí)候,JIT幫我們做的優(yōu)化,它會減少鎖的獲取和釋放次數(shù)。 比如:
while(){
synchronized(){
// 多次的獲取和釋放,成本太高,會被優(yōu)化為下面這種
}
}
synchronized(){
while(){
// 拿到鎖后執(zhí)行循環(huán),只加鎖和釋放一次
}
}鎖消除
鎖消除則是在一個(gè)加鎖的同步代碼塊中,沒有任何共享資源,也不存在鎖競爭的情況,JIT編譯時(shí),就直接將鎖的指令優(yōu)化掉。 比如
synchronized(){
int a = 1;
a++;
//操作局部變量的邏輯
}到此這篇關(guān)于深入詳解Java中synchronized鎖升級的套路的文章就介紹到這了,更多相關(guān)Java synchronized鎖升級內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java編程學(xué)習(xí)的幾個(gè)典型實(shí)例詳解
這篇文章主要給大家介紹了Java編程學(xué)習(xí)的幾個(gè)典型實(shí)例,其中包括模擬酒店房間管理系統(tǒng)、螺旋矩陣 例或者百雞問題的變形等經(jīng)典實(shí)例,具體來一起看詳細(xì)內(nèi)容吧,需要的朋友可以參考學(xué)習(xí)。2017-02-02
springboot項(xiàng)目不同環(huán)境的配置讀取方式
SpringBoot支持application.properties、application.yml、application.yaml三種配置文件類型,可同時(shí)存在并合并配置,配置文件的讀取優(yōu)先級為:application.properties > application.yml > application.yaml,不同位置的相同類型配置文件2024-11-11
關(guān)于Java實(shí)體類Serializable序列化接口的作用和必要性解析
序列化是將對象狀態(tài)轉(zhuǎn)化為可保持或者傳輸?shù)母袷竭^程,與序列化相反的是反序列化,完成序列化和反序列化,可以存儲或傳輸數(shù)據(jù),一般情況下,在定義實(shí)體類時(shí)會使用Serializable,需要的朋友可以參考下2023-05-05
Spring為singleton?bean注入prototype?bean
這篇文章主要介紹了Spring為singleton?bean注入prototype?bean,文章圍繞主題展開詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的小伙伴可以參考一下2022-07-07
java如何對map進(jìn)行排序詳解(map集合的使用)
這篇文章主要介紹了java如何對map進(jìn)行排序,java map集合的使用詳解,大家可以參考使用2013-12-12
Spring?Boot?3?整合?MinIO?實(shí)現(xiàn)分布式文件存儲的全過程
本文介紹了如何使用SpringBoot3和MinIO實(shí)現(xiàn)分布式文件存儲,通過MinIO的分布式對象存儲系統(tǒng),可以解決傳統(tǒng)單機(jī)文件存儲方案在面對大規(guī)模數(shù)據(jù)和高并發(fā)訪問時(shí)的不足,文章詳細(xì)講解了MinIO的安裝、配置和使用,感興趣的朋友一起看看吧2025-03-03
使用java的HttpClient實(shí)現(xiàn)多線程并發(fā)
這篇文章主要介紹了使用java的HttpClient實(shí)現(xiàn)多線程并發(fā)的相關(guān)資料,需要的朋友可以參考下2016-09-09

