Java中的synchronized關鍵字
1、synchronized鎖的底層實現(xiàn)原理
JVM基于進入和退出Monitor
對象來實現(xiàn)方法同步和代碼塊同步。代碼塊同步是使用monitorenter
和monitorexit
指令實現(xiàn)的,monitorenter
指令是在編譯后插入到同步代碼塊的開始位置,而monitorexit
是插入到方法結束處和異常處。任何對象都有一個monitor
與之關聯(lián),當且一個monitor
被持有后,它將處于鎖定狀態(tài)。
根據(jù)虛擬機規(guī)范的要求,在執(zhí)行monitorenter
指令時,首先要去嘗試獲取對象的鎖,如果這個對象沒被鎖定,或者當前線程已經(jīng)擁有了那個對象的鎖,把鎖的計數(shù)器加1;相應地,在執(zhí)行monitorexit
指令時會將鎖計數(shù)器減1,當計數(shù)器被減到0時,鎖就釋放了。如果獲取對象鎖失敗了,那當前線程就要阻塞等待,直到對象鎖被另一個線程釋放為止。
如何判斷這個對象是否被鎖定?對象頭中的MarkWord
字段記錄了該對象的鎖信息。
2、基于synchronized實現(xiàn)單例模式
public class Singleton { private volatile static Singleton uniqueInstance; private Singleton(){ } public static Singleton getUniqueInstance() { //沒有實例化才加鎖 if (uniqueInstance == null) { //給類對象加鎖 synchronized (Singleton.class) { if (uniqueInstance == null) uniqueInstance = new Singleton(); } } return uniqueInstance; } // public static synchronized Singleton getUniqueInstance(){ // if(uniqueInstance==null){ // uniqueInstance = new Singleton(); // } // return uniqueInstance; // } }
首先說一下為什么不采用第二種方式實現(xiàn)單例:不管該對象是否已經(jīng)實例化,都要調(diào)用這個同步方法,會導致大量的線程進入阻塞;而采用雙重鎖檢驗,可以在第一次判斷不為空的時候就直接返回,不用進入同步代碼塊。
幾個要點:
- 為什么uniqueInstance屬性要用volatile修飾?new操作并非一個原子性操作,分為三個步驟(分配對象的內(nèi)存空間、初始化對象、設置
uniqueInstance
指向剛分配的內(nèi)存地址),如果不使用volatile
,2和3之間可能發(fā)生指令重排,導致外部訪問到一個還沒有初始化的對象。 - 為什么構造方法時私有的?防止對象在其他地方被創(chuàng)建。
- 為什么uniqueInstance是私有靜態(tài)的?私有使得外部只能通過特定的方式去訪問對象,靜態(tài)是因為要在靜態(tài)方法中訪問該對象。
- 為什么getUniqueInstance()方式是公共、靜態(tài)的?public使得外界統(tǒng)一通過訪問該方法獲得對象,static使得程序可以通過類名獲取對象。
- 為什么采用雙重檢測初始化對象?第一次檢測主要用來判斷對象是否已經(jīng)創(chuàng)建,如果已創(chuàng)建則直接返回;第二次檢測是因為:可能有多個線程在第一次檢測中發(fā)現(xiàn)對象為空,同時進入同步代碼塊,但只有一個線程會搶到鎖并創(chuàng)建對象,其他線程阻塞排隊等待鎖的釋放。當創(chuàng)建對象的線程返回后,阻塞的線程會被喚醒,這時候對象已經(jīng)不為空,所以需要第二次檢測來阻止對象的多次創(chuàng)建。
3、利用類加載實現(xiàn)單例模式(餓漢模式)
public class SingleTon2 { /** 內(nèi)置對象是靜態(tài)的,并且直接創(chuàng)建對象,保證對象在初始化時加載完成 */ public static SingleTon2 instance = new SingleTon2(); /** 構造方法私有,防止對象在其他地方被創(chuàng)建 */ private SingleTon2(){ } /** 公共靜態(tài)方法返回對象 */ public static SingleTon2 getInstance(){ return instance; } }
到此這篇關于Java中的synchronized關鍵字的文章就介紹到這了,更多相關synchronized關鍵字內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
Jmeter多臺機器并發(fā)請求實現(xiàn)壓力性能測試
這篇文章主要介紹了Jmeter多臺機器并發(fā)請求實現(xiàn)壓力性能測試,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2020-10-10SpringBoot異常: nested exception is java.lang.NoClassDefFoundE
這篇文章主要介紹了SpringBoot異常: nested exception is java.lang.NoClassDefFoundError: javax/servlet/ServletContext解決方案,說明了錯誤原因和解決方案,需要的朋友可以參考下2021-06-06Spring Cloud Gateway 如何修改HTTP響應信息
這篇文章主要介紹了Spring Cloud Gateway 修改HTTP響應信息的方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-07-07Spring中的@Qualifier注解和@Resource注解區(qū)別解析
這篇文章主要介紹了Spring中的@Qualifier注解和@Resource注解區(qū)別解析,@Qualifier注解的用處是當一個接口有多個實現(xiàn)的時候,為了指名具體調(diào)用哪個類的實現(xiàn),@Resource注解可以通過 byName命名和byType類型的方式注入,需要的朋友可以參考下2023-11-11springMVC自定義注解,用AOP來實現(xiàn)日志記錄的方法
下面小編就為大家分享一篇springMVC自定義注解,用AOP來實現(xiàn)日志記錄的方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-01-01java javax.annotation.Resource注解的詳解
這篇文章主要介紹了javax.annotation.Resource注解的詳解的相關資料,需要的朋友可以參考下2016-10-10