synchronize實(shí)現(xiàn)原理

一、基礎(chǔ)概念
1、鎖定義
  如果某一個資源被多個線程共享,為了避免因?yàn)橘Y源搶占導(dǎo)致資源數(shù)據(jù)錯亂,我們需要對線程進(jìn)行同步,那么synchronized就是實(shí)現(xiàn)線程同步的關(guān)鍵字。

2、鎖特性(線程3要素(除可重復(fù)性))

  • 原子性:一個操作要么全部發(fā)生,要么全部不發(fā)生,i++不具備原子性。
  • 可見性:加鎖,釋放鎖之前會將對變量的修改刷新到主存當(dāng)中,或直接操作主存。
  • 有序性:有序性指程序執(zhí)行的順序按照代碼先后執(zhí)行。
  • 可重復(fù)性:一個線程擁有了鎖仍然還可以重復(fù)申請?jiān)撴i。

3、鎖類型
  synchronized可以修飾方法、代碼塊,但是歸根結(jié)底它上鎖的資源只有兩類:一個是對象,一個是類
類上鎖:同步塊、靜態(tài)方法
對象上鎖:同步塊、成員方法

二、鎖實(shí)現(xiàn)原理
1、測試類

/**
 * 歸根結(jié)底它上鎖的資源只有兩類:一個是對象,一個是類。
 */
public class SynchronizedTest {

    public static int i=0;

    SynchronizedTest test = new SynchronizedTest();
    /**
     * 對成員函數(shù)加鎖,必須獲得該類的實(shí)例對象的鎖才能進(jìn)入同步塊
     */
    public synchronized void test1() {
        i++;
    }

    /**
     * 對靜態(tài)方法加鎖,必須獲得該類的鎖才能進(jìn)入同步塊,本文暫不關(guān)注該方法
     */
    public static synchronized void test2() {
        System.out.println("Hello static test2");
    }

    public void test3() {
        /**
         * 必須獲得類鎖
         */
        synchronized(SynchronizedTest.class) {
            System.out.println("Hello test3");
        }

        /**
         * 必須獲得對象鎖, 本文暫不關(guān)注該代碼塊
         */
        synchronized(test) {
            System.out.println("Hello test3-2");
        }
    }
}

2、查看字節(jié)碼文件

// 編譯生成字節(jié)碼文件
javac SynchronizedTest.java 
// 反編譯查看字節(jié)碼文件
javap -verbose SynchronizedTest.class

3、同步塊的實(shí)現(xiàn)


同步塊字節(jié)碼

由monitorenter指令進(jìn)入,然后monitorexit釋放鎖,在執(zhí)行monitorenter之前需要嘗試獲取鎖,如果這個對象沒有被鎖定,或者當(dāng)前線程已經(jīng)擁有了這個對象的鎖,那么就把鎖的計(jì)數(shù)器加1。當(dāng)執(zhí)行monitorexit指令時,鎖的計(jì)數(shù)器也會減1。

4、同步方法


同步方法

flags里面多了一個ACC_SYNCHRONIZED標(biāo)志,這標(biāo)志用來告訴JVM這是一個同步方法,在進(jìn)入該方法之前先獲取相應(yīng)的鎖,鎖的計(jì)數(shù)器加1,方法結(jié)束后計(jì)數(shù)器-1,如果獲取失敗就阻塞住,直到該鎖被釋放。

三、鎖底層實(shí)現(xiàn)原理
  鎖存放在對象頭的Mark Word區(qū)域中。
  鎖總共四個狀態(tài):無鎖狀態(tài)、偏向鎖、輕量級鎖、重量級鎖,其中無鎖就是一種狀態(tài)了。鎖的類型和狀態(tài)在對象頭Mark Word中都有記錄。
  每個對象都存在著一個monitor與之關(guān)聯(lián),當(dāng)線程試圖獲取對象鎖時自動生成,但當(dāng)一個monitor被某個線程持有后,它便處于鎖定狀態(tài),Mark Word會記錄鎖定狀態(tài)和線程id,monitor中的_owner標(biāo)志會指向持有monitor的線程同時monitor中的計(jì)數(shù)器count加1。
總結(jié):
  對象頭有Mark Word區(qū)域,當(dāng)線程視圖獲取對象鎖的時候,會生成一個monitor與對象關(guān)聯(lián)。線程獲取鎖其實(shí)就是獲取對象的monitor;當(dāng)一個monitor被線程持有以后,monitor中的_owner標(biāo)志會指向持有monitor的線程,同時monitor中的計(jì)數(shù)器count加1。Mark Word區(qū)域會記錄鎖定狀態(tài)、鎖的類型和線程id。

四、鎖優(yōu)化
鎖升級方向:無鎖——>偏向鎖——>輕量級鎖——>重量級鎖,并且方向不可逆。

1、偏向鎖:
  在大多數(shù)情況下,鎖不存在多線程競爭,總是由同一線程多次獲得,那么此時就是偏向鎖。
  如果一個線程獲得了鎖,那么鎖就進(jìn)入偏向模式,此時Mark Word的結(jié)構(gòu)也就變?yōu)槠蜴i結(jié)構(gòu),當(dāng)該線程再次請求鎖時,無需再做任何同步操作,即獲取鎖的過程只需要檢查Mark Word的鎖標(biāo)記位為偏向鎖以及當(dāng)前線程ID等于Mark Word的ThreadID即可,這樣就省去了大量有關(guān)鎖申請的操作。

2、輕量級鎖:
輕量級鎖是由偏向鎖升級而來,當(dāng)存在第二個線程申請同一個鎖對象時,偏向鎖就會立即升級為輕量級鎖。

3、重量級鎖:
重量級鎖是由輕量級鎖升級而來,當(dāng)同一時間有多個線程競爭鎖時,鎖就會被升級成重量級鎖,此時其申請鎖帶來的開銷也就變大。重量級鎖一般使用場景會在追求吞吐量,同步塊或者同步方法執(zhí)行時間較長的場景。

4、鎖消除:
  消除鎖是虛擬機(jī)另外一種鎖的優(yōu)化,這種優(yōu)化更徹底,在JIT編譯時,對運(yùn)行上下文進(jìn)行掃描,去除不可能存在競爭的鎖。比如局部變量的加鎖。

5、鎖粗化:
  鎖粗化是虛擬機(jī)對另一種極端情況的優(yōu)化處理,通過擴(kuò)大鎖的范圍,避免反復(fù)加鎖和釋放鎖。比如把for循環(huán)內(nèi)部的鎖擴(kuò)大到for循環(huán)外部。

6、自旋鎖與自適應(yīng)自旋鎖
  輕量級鎖失敗后,虛擬機(jī)為了避免線程真實(shí)地在操作系統(tǒng)層面掛起,還會進(jìn)行一項(xiàng)稱為自旋鎖的優(yōu)化手段。

6.1、自旋鎖
  許多情況下,共享數(shù)據(jù)的鎖定狀態(tài)持續(xù)時間較短,切換線程不值得,通過讓線程執(zhí)行循環(huán)等待鎖的釋放,不讓出CPU。如果得到鎖,就順利進(jìn)入臨界區(qū)。如果還不能獲得鎖,那就會將線程在操作系統(tǒng)層面掛起,這就是自旋鎖的優(yōu)化方式。但是它也存在缺點(diǎn):如果鎖被其他線程長時間占用,一直不釋放CPU,會帶來許多的性能開銷。

6.2、自適應(yīng)自旋鎖
  這種相當(dāng)于是對上面自旋鎖優(yōu)化方式的進(jìn)一步優(yōu)化,它的自旋的次數(shù)不再固定,其自旋的次數(shù)由前一次在同一個鎖上自旋并成功獲取鎖的擁有者的自旋次數(shù)來決定,這就解決了自旋鎖帶來的缺點(diǎn)。

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