摘要:當(dāng)前線程使用將對象頭的替換為鎖記錄指針,如果成功,當(dāng)前線程獲得鎖如果失敗,表示其他線程競爭鎖,當(dāng)前線程便嘗試使用自旋來獲取鎖。重量級鎖是悲觀鎖的一種,自旋鎖輕量級鎖與偏向鎖屬于樂觀鎖。
操作系統(tǒng)在面對線程間同步的時候,會支持例如semaphore信號量和mutex互斥量等同步原語,而monitor是在編程語言中被實現(xiàn)的,下面介紹一下java中monitor(監(jiān)視器/管程:管理共享變量以及對其的操作過程,讓他們支持并發(fā))的實現(xiàn)原理:
以一個阻塞隊列的實現(xiàn)來舉例:
同時,java內(nèi)置的synchronized關(guān)鍵字可以認(rèn)為是MESA模型的簡化版,其只能有一個條件變量,但編譯器會自動添加加鎖與解鎖的代碼。synchronized關(guān)鍵字可以修飾實例方法、類方法以及代碼塊,如果修飾的是代碼塊,需要制定關(guān)聯(lián)的Object;如果修飾的是實例方法,那么其關(guān)聯(lián)的對象實際上是this;如果修飾的是類方法,那么其關(guān)聯(lián)的對象是this.class。這些關(guān)聯(lián)的對象就是MESA模型里的條件變量。
synchronized實現(xiàn)原理JVM基于進(jìn)入和退出monitor對象來實現(xiàn)同步,同步代碼塊采用添加moniterenter、moniterexit,同步方法使用ACC_SYNCHRONIZED標(biāo)記符隱式實現(xiàn)。每個對象都有一個monitor與之關(guān)聯(lián),運(yùn)行到moniterenter時嘗試獲取對應(yīng)monitor的所有權(quán),獲取成功就將monitor的進(jìn)入數(shù)加1(所以是可重入鎖,也被稱為重量級鎖),否則就阻塞,擁有monitor的線程運(yùn)行到moniterexit時進(jìn)入數(shù)減1,為0時釋放monitor。
java中每個對象都有一個對象頭,synchronized所用的鎖就是存在對象頭里的。如果是非數(shù)組的對象是8個字節(jié)(32位JVM)或者16字節(jié)(64位JVM),數(shù)組對象還會有一個數(shù)組長度(4個字節(jié))。以32位JVM非數(shù)組對象為例:
鎖信息就存在前4個字節(jié)的MarkWord中,JVM對synchronized的加鎖過程優(yōu)化為:
檢測Mark Word里面是不是當(dāng)前線程的ID,如果是,表示當(dāng)前線程處于偏向鎖
如果不是,則使用CAS將當(dāng)前線程的ID替換Mard Word,如果成功則表示當(dāng)前線程獲得偏向鎖,置偏向標(biāo)志位1
如果失敗,則說明發(fā)生競爭,撤銷偏向鎖,進(jìn)而升級為輕量級鎖。
當(dāng)前線程使用CAS將對象頭的Mark Word替換為鎖記錄指針,如果成功,當(dāng)前線程獲得鎖
如果失敗,表示其他線程競爭鎖,當(dāng)前線程便嘗試使用自旋來獲取鎖。
如果自旋成功則依然處于輕量級狀態(tài)。
如果自旋失敗,則升級為重量級鎖。
重量級鎖是悲觀鎖的一種,自旋鎖、輕量級鎖與偏向鎖屬于樂觀鎖。
CAS設(shè)計讀取-比較-寫入三個操作,是在CPU指令層面保證其原子性,volatile是保證多線程下的內(nèi)存可見性,二者需配合使用。另外還需注意CPU緩存行(一次以32/64字節(jié)為單位從主內(nèi)存中讀取數(shù)據(jù)到緩存)包含多個變量所帶來的隱形同步問題:其中一個變量被volatile修飾,導(dǎo)致另外一個變量在另一個CPU核上(另一個線程)的讀寫也要被強(qiáng)制刷新緩存。
管程:并發(fā)編程的萬能鑰匙
java 中的鎖 -- 偏向鎖、輕量級鎖、自旋鎖、重量級鎖
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/77770.html
摘要:語言在之前,提供的唯一的并發(fā)原語就是管程,而且之后提供的并發(fā)包,也是以管程技術(shù)為基礎(chǔ)的。但是管程更容易使用,所以選擇了管程。線程進(jìn)入條件變量的等待隊列后,是允許其他線程進(jìn)入管程的。并發(fā)編程里兩大核心問題互斥和同步,都可以由管程來幫你解決。 并發(fā)編程這個技術(shù)領(lǐng)域已經(jīng)發(fā)展了半個世紀(jì)了。有沒有一種核心技術(shù)可以很方便地解決我們的并發(fā)問題呢?這個問題, 我會選擇 Monitor(管程)技術(shù)。Ja...
摘要:請參看前一篇文章并發(fā)學(xué)習(xí)筆記一原子性可見性有序性問題六等待通知機(jī)制什么是等待通知機(jī)制當(dāng)線程不滿足某個條件,則進(jìn)入等待狀態(tài)如果線程滿足要求的某個條件后,則通知等待的線程重新執(zhí)行。經(jīng)極客時間并發(fā)編程實戰(zhàn)專欄內(nèi)容學(xué)習(xí)整理 請參看前一篇文章:Java 并發(fā)學(xué)習(xí)筆記(一)——原子性、可見性、有序性問題 六、等待—通知機(jī)制 什么是等待通知—機(jī)制?當(dāng)線程不滿足某個條件,則進(jìn)入等待狀態(tài);如果線程滿足要...
摘要:基本元素機(jī)制需要幾個元素來配合,分別是臨界區(qū)對象及鎖條件變量以及定義在對象上的,操作。這個外部條件在機(jī)制中稱為條件變量。提供的機(jī)制,其實是,等元素合作形成的,甚至說外部的條件變量也是個組成部分。 monitor的概念 管程,英文是 Monitor,也常被翻譯為監(jiān)視器,monitor 不管是翻譯為管程還是監(jiān)視器,都是比較晦澀的,通過翻譯后的中文,并無法對 monitor 達(dá)到一個直觀的描...
摘要:那并發(fā)里面的理論和模型是什么呢那便要從操作系統(tǒng)中解決并發(fā)問題的一種模型管程講起了。當(dāng)一個進(jìn)程使用完管程后,它必須釋放管程并喚醒等待管程的某一個進(jìn)程。總結(jié)在并發(fā)編程領(lǐng)域,有兩大核心問題互斥和同步,而這兩個問題,管程模型都可以解決。 為什么需要了解管程 Java并發(fā)編程是Java中高級程序員必備的一項技能,但是真正學(xué)明白并發(fā)編程也并非易事。正如Java并發(fā)編程實踐中的一句話編寫正確的程序并...
摘要:由此可見,自旋鎖和各有優(yōu)劣,他們分別適用于競爭不多和競爭激烈的場景中。每一個試圖進(jìn)入同步代碼塊的線程都會被封裝成對象,它們或在對象的中,或在中,等待成為對象的成為的對象即獲取了監(jiān)視器鎖。 前言 系列文章目錄 前面兩篇文章我們介紹了synchronized同步代碼塊以及wait和notify機(jī)制,大致知道了這些關(guān)鍵字和方法是干什么的,以及怎么用。 但是,知其然,并不知其所以然。 例如...
閱讀 3685·2021-11-23 09:51
閱讀 1682·2021-10-22 09:53
閱讀 1361·2021-10-09 09:56
閱讀 867·2019-08-30 13:47
閱讀 2165·2019-08-30 12:55
閱讀 1609·2019-08-30 12:46
閱讀 1122·2019-08-30 10:51
閱讀 2420·2019-08-29 12:43