淺析Java關(guān)鍵詞synchronized的使用
1 引入Synchronized
- Synchronized是java虛擬機為線程安全而引入的。
- 互斥同步是一種最常見的并發(fā)正確性的保障手段。同步是指在多個線程并發(fā)訪問共享數(shù)據(jù)時,保證共享數(shù)據(jù)在同一個時刻只被一條線程使用。
- synchronized是最基本的互斥同步手段,它是一種塊結(jié)構(gòu)的同步語法。
- synchronized修飾代碼塊,無論該代碼塊正常執(zhí)行完成還是發(fā)生異常,都會釋放鎖
synchronized對線程訪問的影響:
- 被synchronized修飾的同步塊在持有鎖的線程執(zhí)行完畢并釋放鎖之前,會阻塞其他線程的進入。
- 被synchronized修飾的同步塊對同一條線程是可重入的
2 Synchronized的使用
可以作用在方法上或者方法里的代碼塊:
- 修飾方法,包括實例方法和靜態(tài)方法
- 修飾方法里的代碼塊,這時需要一個引用作為參數(shù)。
- Synchronized作用地方不同,產(chǎn)生的鎖類型也不同,分為對象鎖和類鎖
2.1 對象鎖
Synchronized修飾實例方法或者代碼塊(鎖對象不是*.class),此時生產(chǎn)對象鎖。多線程訪問該類的同一個對象的sychronized塊是同步的,訪問不同對象不受同步限制。
2.1.1 Synchronized修飾實例方法
public static void main(String[] args){ TempTest tempTest = new TempTest(); Thread t1 = new Thread(() -> { tempTest.doing(Thread.currentThread().getName()); }); Thread t2 = new Thread(() -> { tempTest.doing(Thread.currentThread().getName()); }); t1.start(); t2.start(); } //同一時刻只能被一個線程調(diào)用 private synchronized void doing(String threadName){ for(int i=0;i<3;i++){ System.out.println("current thread is : "+threadName); try { Thread.sleep(50); } catch (InterruptedException e) {} } }
運行結(jié)果:
current thread is : Thread-0
current thread is : Thread-0
current thread is : Thread-0
current thread is : Thread-1
current thread is : Thread-1
current thread is : Thread-1
2.1.2 Synchronized修飾代碼塊
public class SynchronizedObjectLock implements Runnable { static SynchronizedObjectLock instence = new SynchronizedObjectLock(); @Override public void run() { // 同步代碼塊形式:鎖為this,兩個線程使用的鎖是一樣的,線程1必須要等到線程0釋放了該鎖后,才能執(zhí)行 synchronized (this) { System.out.println("我是線程" + Thread.currentThread().getName()); try { Thread.sleep(3000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(Thread.currentThread().getName() + "結(jié)束"); } } public static void main(String[] args) { Thread t1 = new Thread(instence); Thread t2 = new Thread(instence); t1.start(); t2.start(); } }
運行結(jié)果:
我是線程Thread-0
Thread-0結(jié)束
我是線程Thread-1
Thread-1結(jié)束
2.2 類鎖
synchronize修飾靜態(tài)方法或指定鎖對象為Class,此時產(chǎn)生類鎖。多線程訪問該類的所有對象的sychronized塊是同步的,
2.2.1 synchronize修飾靜態(tài)方法
public static void main(String[] args){ TempTest tempTest1 = new TempTest(); TempTest tempTest2 = new TempTest(); //雖然創(chuàng)建了兩個TempTest實例,但是依然是調(diào)用同一個doing方法(因為是個static);因此doing還是會依次執(zhí)行 Thread t1 = new Thread(() -> tempTest1.doing(Thread.currentThread().getName())); Thread t2 = new Thread(() -> tempTest2.doing(Thread.currentThread().getName())); t1.start(); t2.start(); } //修飾靜態(tài)方法,則是類鎖; private static synchronized void doing(String threadName){ for(int i=0;i<3;i++){ System.out.println("current thread is : "+threadName); try { Thread.sleep(50); } catch (InterruptedException e) {} } }
運行結(jié)果:有序輸出 【如果去掉static ,則線程會交替執(zhí)行doing】
current thread is : Thread-0
current thread is : Thread-0
current thread is : Thread-0
current thread is : Thread-1
current thread is : Thread-1
current thread is : Thread-1
2.2.2 synchronize指定鎖對象為Class
public class SynchronizedObjectLock implements Runnable { static SynchronizedObjectLock instence1 = new SynchronizedObjectLock(); static SynchronizedObjectLock instence2 = new SynchronizedObjectLock(); @Override public void run() { // 所有線程需要的鎖都是同一把 synchronized(SynchronizedObjectLock.class){ System.out.println("我是線程" + Thread.currentThread().getName()); try { Thread.sleep(3000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(Thread.currentThread().getName() + "結(jié)束"); } } public static void main(String[] args) { Thread t1 = new Thread(instence1); Thread t2 = new Thread(instence2); t1.start(); t2.start(); } }
結(jié)果:
我是線程Thread-0
Thread-0結(jié)束
我是線程Thread-1
Thread-1結(jié)束
3 Synchronized原理分析
3.1 虛擬機如何辨別和處理synchronized
虛擬機可以從常量池中的方法表結(jié)構(gòu)中的ACC_ SYNCHRONIZED訪問標(biāo)志區(qū)分一個方法是否是同步方法。
當(dāng)調(diào)用方法時,調(diào)用指令將會檢查方法的ACC_ SYNCHRONIZED訪問標(biāo)志是否設(shè)置,如果設(shè)置了,執(zhí)行線程將先持有同步鎖,然后執(zhí)行方法,最后在方法完成時釋放同步鎖。
在方法執(zhí)行期間,執(zhí)行線程持有了同步鎖,其他任何線程都無法再獲得同一個鎖。
如果一個同步方法執(zhí)行期間拋出了異常,并且在方法內(nèi)部無法處理此異常,那這個同步方法所持有的鎖將在異常拋到同步方法之外時自動釋放。
3.2 虛擬機對synchronized的編譯處理
以下代碼:
public class Foo { void onlyMe(Foo f) { synchronized(f) { doSomething(); } } private void doSomething(){ } }
編譯后,這段代碼生成的字節(jié)碼序列如下:
- synchronized關(guān)鍵字經(jīng)過Javac編譯之后,會在同步塊的前后生成monitorenter和monitorexit兩個字節(jié)碼指令。
- 指令含義:monitorenter:獲取對象的鎖;monitorexit:釋放對象的鎖
- 執(zhí)行monitorenter指令時,首先嘗試獲取對象的鎖。如果對象沒被鎖定,或者當(dāng)前線程已經(jīng)持有了對象的鎖,就把鎖的計數(shù)器的值增加1
- 執(zhí)行monitorexit指令時,將鎖計數(shù)器的值減1,一旦計數(shù)器的值為零,鎖隨即就被釋放
- 如果獲取對象鎖失敗,那當(dāng)前線程阻塞等待,直到鎖被釋放。
- 為了保證在方法異常完成時monitorenter和monitorexit指令依然可以正確配對執(zhí)行,編譯器會自動產(chǎn)生一個異常處理程序,它的目的就是用來執(zhí)行monitorexit指令。
3.3 虛擬機執(zhí)行加鎖和釋放鎖的過程
那么重點來了到這里,有幾個問題需明確:
- 什么叫對象的鎖?
- 如何確定鎖被線程持有?
- 執(zhí)行monitorenter后,對象發(fā)生什么變化?
- 鎖計數(shù)值保存在哪里,如何獲取到?
1. 什么叫對象的鎖?對象的內(nèi)存結(jié)構(gòu)參考文末補充內(nèi)容
- 鎖,一種可以被讀寫的資源,對象的鎖是對象的一部分。
- 對象的結(jié)構(gòu)中有部分稱為對象頭。
- 對象頭中有2bit空間,用于存儲鎖標(biāo)志,通過該標(biāo)志位來標(biāo)識對象是否被鎖定。
2. 如果確定鎖被線程持有?
- 代碼即將進入同步塊的時,如果鎖標(biāo)志位為“01”(對象未被鎖定),虛擬機首先將在當(dāng)前線程的棧幀中建立一個名為鎖記錄的空間,存儲鎖對象Mark Word的拷貝。(線程開辟空間并存儲對象頭)
- 虛擬機將使用CAS操作嘗試把對象的Mark Word更新成指向鎖記錄的指針(對象頭的mw存儲指向線程“鎖記錄”中的指針)
- 如果CAS操作成功,即代表該線程擁有了這個對象的鎖,并且將對象的鎖標(biāo)志位轉(zhuǎn)變?yōu)?ldquo;00”
- 如果CAS操作失敗,那就意味著至少存在一條線程與當(dāng)前線程競爭獲取該對象的鎖。虛擬機首先會檢查對象的Mark Word是否指向當(dāng)前線程的棧幀,如果是,說明當(dāng)前線程已經(jīng)擁有了這個對象的鎖,那直接進入同步塊繼續(xù)執(zhí)行,否則就說明這個鎖對象已經(jīng)被其他線程搶占。
- 解鎖過程:CAS操作把線程中保存的MW拷貝替換回對象頭中。假如能夠成功替換,那整個同步過程就順利完成了;如果替換失敗,則說明有其他線程嘗試過獲取該鎖,就要在釋放鎖的同時,喚醒被掛起的線程。
3 執(zhí)行monitorenter后,對象發(fā)生什么變化?
- 對象的鎖標(biāo)志位轉(zhuǎn)變?yōu)?ldquo;00”
- 擁有對象鎖的線程開辟了新空間,保存了對象的Mark Word信息
- 對象的Mark Word保存了線程的鎖記錄空間的地址拷貝
4 鎖計數(shù)值保存在哪里?
我還沒搞懂。
monitorenter指令執(zhí)行的過程:
4 Synchronized與Lock
synchronized的缺陷
- 在多線程競爭鎖時,當(dāng)一個線程獲取鎖時,它會阻塞所有正在競爭的線程,這樣對性能帶來了極大的影響。
- 掛起線程和恢復(fù)線程的操作都需要轉(zhuǎn)入內(nèi)核態(tài)中完成,上下文切換需要消耗很大性能。
- 效率低:鎖的釋放情況少,只有代碼執(zhí)行完畢或者異常結(jié)束才會釋放鎖;試圖獲取鎖的時候不能設(shè)定超時,不能中斷一個正在使用鎖的線程,相對而言,Lock可以中斷和設(shè)置超時
- 不夠靈活:加鎖和釋放的時機單一,每個鎖僅有一個單一的條件(某個對象),相對而言,讀寫鎖更加靈活
5 使用Synchronized有哪些要注意的
鎖對象不能為空,因為鎖的信息都保存在對象頭里
作用域不宜過大,影響程序執(zhí)行的速度,控制范圍過大,編寫代碼也容易出錯
在能選擇的情況下,既不要用Lock也不要用synchronized關(guān)鍵字,用java.util.concurrent包中的各種各樣的類,如果有必要,使用synchronized關(guān)鍵,因為代碼量少,避免出錯
synchronized實際上是非公平的,新來的線程有可能立即獲得執(zhí)行,而在等待區(qū)中等候已久的線程可能再次等待,這樣有利于提高性能,但是也可能會導(dǎo)致饑餓現(xiàn)象。
知識補充
Java內(nèi)存層面的對象認識
說明:此分析基于HotSpot虛擬機
1 對象的創(chuàng)建
Java對象的創(chuàng)建方式有三種:
- 通過new創(chuàng)建
- 通過反序列化創(chuàng)建
- 通過復(fù)制創(chuàng)建
通過new方式的對象創(chuàng)建過程如下:
創(chuàng)建過程說明:
- 執(zhí)行字節(jié)碼遇到new指令時,首先將去檢查這個指令的參數(shù)是否能在常量池中定位到 一個類的符號引用。
- 類的初始化過程在后續(xù)章節(jié)詳細補充
- 給對象分配初始內(nèi)存空間有兩種方式:指針碰撞 和 空閑列表。
- 分配空間后,清空該段的【不包括對象頭】值,保證對象屬性的不設(shè)值就使用初始值
- 對象頭信息包括:元數(shù)據(jù)信息、對象的哈希碼、對象的GC分代年齡
- 執(zhí)行構(gòu)造函數(shù),給屬性初始化設(shè)置的值
2 對象的內(nèi)存布局
對象存儲的內(nèi)容分類以及明細如下:
關(guān)于對象頭的補充說明:
- 對象頭,在字長為32位和64位的虛擬機中分別為32比特(4字節(jié))64比特(8字節(jié))。
- 對象頭的類型指針:不一定所有對象都會存儲這項信息,意味著訪問對象所屬的類不一定通過對象自身。
- 如果對象是一個Java數(shù)組,那在對象頭中還必須有一塊用于記錄數(shù)組長度的數(shù)據(jù)。
關(guān)于實例數(shù)據(jù)的補充說明:
- 對象屬性的存儲順序,受到虛擬機分配策略參數(shù)(-XX:FieldsAllocationStyle參數(shù))和字段在Java源碼中定義順序的影響
- 默認的分配策略下:占據(jù)相同字節(jié)數(shù)的屬性會排列在一起,滿足該條件下,父類的屬性排在前面。
關(guān)于對齊填充的說明:
不一定會存在,因為對象的大小一定是8字節(jié)的整數(shù)倍,因此需要對齊填充這部分,充當(dāng)占位符
在32位字長的虛擬機下,對象的內(nèi)存分布情況如下:
3 對象的訪問定位
對象訪問方式也是由虛擬機實現(xiàn)而定的,主流的訪問方式主要有使用句柄和直接指針兩種:
3.1句柄訪問
說明:
句柄訪問方式,Java堆中將可能會劃分出一塊內(nèi)存來作為句柄池,reference中存儲的就是對象的句柄地址,而句柄中包含了對象實例數(shù)據(jù)與類型數(shù)據(jù)各自具體的地址
3.2 直接指針訪問
說明:
使用直接指針來訪問最大的好處就是速度更快,只需要一次定位就能找到實例數(shù)據(jù),而句柄池則需要兩次:(需要先定位句柄池,再定位實例數(shù)據(jù))
以上就是淺析Java關(guān)鍵詞synchronized的使用的詳細內(nèi)容,更多關(guān)于Java關(guān)鍵詞synchronized的資料請關(guān)注腳本之家其它相關(guān)文章!
- Java同步鎖Synchronized底層源碼和原理剖析(推薦)
- java同步鎖的正確使用方法(必看篇)
- 95%的Java程序員人都用不好Synchronized詳解
- Java?synchronized同步關(guān)鍵字工作原理
- Java synchronized偏向鎖的概念與使用
- Java?synchronized輕量級鎖實現(xiàn)過程淺析
- Java synchronized重量級鎖實現(xiàn)過程淺析
- Java @Transactional與synchronized使用的問題
- Java?synchronized與死鎖深入探究
- Java synchronized與CAS使用方式詳解
- synchronized及JUC顯式locks?使用原理解析
- java鎖synchronized面試常問總結(jié)
- Java?HashTable與Collections.synchronizedMap源碼深入解析
- Java?Synchronized鎖的使用詳解
- AQS加鎖機制Synchronized相似點詳解
- Java必會的Synchronized底層原理剖析
- 一個例子帶你看懂Java中synchronized關(guān)鍵字到底怎么用
- 詳解Java?Synchronized的實現(xiàn)原理
- Synchronized?和?ReentrantLock?的實現(xiàn)原理及區(qū)別
- Java同步鎖synchronized用法的最全總結(jié)
相關(guān)文章
SpringCloud實現(xiàn)灰度發(fā)布的方法步驟
本文主要介紹了SpringCloud實現(xiàn)灰度發(fā)布的方法步驟,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-05-05詳解mybatis多對一關(guān)聯(lián)查詢的方式
這篇文章主要給大家介紹了關(guān)于mybatis多對一關(guān)聯(lián)查詢的相關(guān)資料,文中將關(guān)聯(lián)方式以及配置方式介紹的很詳細,需要的朋友可以參考下2021-06-06Java進行反編譯生成.java文件方式(javap、jad下載安裝使用)
這篇文章主要介紹了Java進行反編譯生成.java文件方式(javap、jad下載安裝使用),具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-12-12springboot將mybatis升級為mybatis-plus的實現(xiàn)
之前項目工程用的是mybatis,現(xiàn)在需要將其替換為mybatis-plus,本文主要介紹了springboot將mybatis升級為mybatis-plus的實現(xiàn),具有一定的參考價值,感興趣的可以了解一下2023-09-09