Java對象級別與類級別的同步鎖synchronized語法示例
Java synchronized 關(guān)鍵字 可以將一個代碼塊或一個方法標(biāo)記為同步代碼塊。同步代碼塊是指同一時間只能有一個線程執(zhí)行的代碼,并且執(zhí)行該代碼的線程持有同步鎖。synchronized
關(guān)鍵字可以作用于
- 一個代碼塊
- 一種方法
當(dāng)一個方法或代碼塊被聲明為synchronized時,如果一個線程正在執(zhí)行該synchronized 方法或代碼塊,其他線程會被阻塞,直到持有同步鎖的線程釋放。根據(jù)鎖定的范圍可以分為
- 類級別的鎖可以防止多個線程在運行時同時進(jìn)入該類所有實例化對象的 synchronized代碼塊中。
- 對象級別的鎖可以防止多個線程在運行時同時進(jìn)入當(dāng)前(或某一個)實例化對象的 synchronized代碼塊中。
1. 對象級別的同步鎖
對象級別的同步鎖:當(dāng)我們想要在多線程環(huán)境下同步執(zhí)行一個非靜態(tài)方法或非靜態(tài)代碼塊時,在類的方法或代碼塊加上synchronized關(guān)鍵字,可以保證對象實例級別數(shù)據(jù)的線程安全。(比較后文的類級別的同步鎖,回頭來理解這句話)
對象級別的加鎖的代碼如下,如:在方法上加鎖,鎖對象為當(dāng)前類的實例化對象
public class DemoClass{ public synchronized void demoMethod(){} }
如:為代碼塊加鎖,鎖對象為this對象
public class DemoClass{ public void demoMethod(){ synchronized (this){ //同步代碼塊 } } }
如:為代碼塊加鎖,鎖對象為我們創(chuàng)建的任意一個對象。不要使用非final的成員變量作為同步鎖對象,因為非final成員變量可以被重新賦值,導(dǎo)致不同的線程使用不同的對象作為鎖,達(dá)不到同步鎖定的效果。
public class DemoClass{ //注意這里的關(guān)鍵字final非常重要,看說明 private final Object lock = new Object(); public void demoMethod(){ synchronized (lock){ //同步代碼塊 } } }
2. 類級別的同步鎖
類級別的鎖可以防止多個線程在運行時進(jìn)入該類所有實例化對象的 "synchronized塊中。也就是說如果運行時有100個DemoClass
的實例,那么每次只有一個線程能夠在任何一個實例中執(zhí)行demoMethod()
,所有其他實例的所有其他線程都被鎖定。
為了保障靜態(tài)數(shù)據(jù)線程安全,應(yīng)該使用類級別的鎖定。我們知道static關(guān)鍵字將方法的數(shù)據(jù)關(guān)聯(lián)到類的級別上,所以在靜態(tài)方法上使用鎖。
靜態(tài)方法加鎖,對該類所有的實例化對象生效
public class DemoClass{ //靜態(tài)方法加鎖,對該類所有的實例化對象生效 public synchronized static void demoMethod(){ } }
獲取 .class類的引用,類級別的鎖
public class DemoClass{ public void demoMethod(){ //獲取 .class類的引用,類級別的鎖,對該類所有的實例化對象生效 synchronized (DemoClass.class){ //同步代碼塊 } } }
使用靜態(tài)對象的鎖,類級別的鎖
public class DemoClass{ //靜態(tài)對象,類級別,注意這里的關(guān)鍵字final非常重要 private final static Object lock = new Object(); public void demoMethod(){ //使用靜態(tài)對象的鎖,類級別鎖,對該類所有的實例化對象生效 synchronized (lock){ //同步代碼塊 } } }
3. 總結(jié)
- Java中的同步機制保證了兩個或多個線程無法同時執(zhí)行一個需要相同同步鎖的方法。
- "synchronized "關(guān)鍵字只能用于方法和代碼塊。這些方法或代碼塊可以是靜態(tài)或非靜態(tài)的。
- 當(dāng)一個線程進(jìn)入synchronized方法或代碼塊時,它就會獲得一個鎖,當(dāng)它離開同步方法或代碼塊時,它就會釋放這個鎖。如果線程執(zhí)行過程出現(xiàn)任何錯誤或異常,鎖也會被釋放。
- 使用"synchronized "關(guān)鍵字持有的鎖在本質(zhì)上是可重入的,這意味著如果一個同步方法調(diào)用另一個使用相同鎖的同步方法,那么持有鎖的當(dāng)前線程可以進(jìn)入該方法而無需再次獲得鎖。
- 如果同步塊中使用的對象為空,Java synchronized 將拋出NullPointerException
- 使用synchronized同步方法會給你的應(yīng)用程序帶來性能成本。因此,盡量在絕對需要的情況下才使用同步。另外優(yōu)先考慮使用同步代碼塊,并且只同步代碼的關(guān)鍵部分。
- 靜態(tài)同步方法和非靜態(tài)同步方法有可能同時或并發(fā)運行,因為它們使用的是不同的鎖。
- 根據(jù)Java語言規(guī)范,你不能在構(gòu)造函數(shù)中使用synchronized關(guān)鍵字。這是不合法的,會導(dǎo)致編譯錯誤。
- 不要使用非final的成員變量作為同步鎖對象,因為非final成員變量可以被重新賦值,導(dǎo)致不同的線程使用不同的對象作為鎖,達(dá)不到同步鎖定的效果。
- 不要使用字符串字面量作為鎖對象,如:String a = "1";,因為它們可能會被應(yīng)用程序中的其他地方引用,并可能導(dǎo)致死鎖。用new關(guān)鍵字創(chuàng)建的字符串對象可以安全使用。
以上就是Java對象級別與類級別的同步鎖synchronized語法示例的詳細(xì)內(nèi)容,更多關(guān)于Java對象與類級別同步鎖synchronized的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
java中l(wèi)ambda(函數(shù)式編程)一行解決foreach循環(huán)問題
這篇文章主要介紹了java中l(wèi)ambda(函數(shù)式編程)一行解決foreach循環(huán)問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-07-07spring boot配置ssl實現(xiàn)HTTPS的方法
這篇文章主要介紹了spring boot配置ssl實現(xiàn)HTTPS的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2019-03-03SpringCloud項目中Feign組件添加請求頭所遇到的坑及解決
這篇文章主要介紹了SpringCloud項目中Feign組件添加請求頭所遇到的坑及解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-04-04selenium+java+chrome環(huán)境搭建的方法步驟
這篇文章主要介紹了selenium+java+chrome環(huán)境搭建的方法步驟,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-07-07SpringBoot如何實現(xiàn)定時任務(wù)示例詳解
使用定時任務(wù)完成一些業(yè)務(wù)邏輯,比如天氣接口的數(shù)據(jù)獲取,定時發(fā)送短信,郵件。以及商城中每天用戶的限額,定時自動收貨等等,這篇文章主要給大家介紹了關(guān)于SpringBoot如何實現(xiàn)定時任務(wù)的相關(guān)資料,需要的朋友可以參考下2021-10-10- String在棧中,StringBuffer在堆中!所以String是不可變的,數(shù)據(jù)是共享的。StringBuffer都是獨占的,是可變的(因為每次都是創(chuàng)建新的對象?。?/div> 2015-11-11
MyBatis-Plus工具使用之EntityWrapper解析
這篇文章主要介紹了MyBatis-Plus工具使用之EntityWrapper解析,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-03-03最新評論