Java中使用synchronized關(guān)鍵字實現(xiàn)簡單同步操作示例
簡單記錄下java中synchronized關(guān)鍵字的使用方法。
在介紹之前需要明確下java中的每一個類的對象實例都有且只有一個鎖(lock)和之相關(guān)聯(lián),synchronized關(guān)鍵字只作用于該鎖,即可以認(rèn)為synchronized只對java類的對象實例起作用。
synchronized修飾函數(shù)
public synchronized aMethod(){
}
這就是最常用的情景,那么這個同步方法的用途是啥,為了方便就記作aMethod方法。
1、synchronized鎖定的是調(diào)用這個同步方法的對象實例,舉個例子,同一個實例P1在不同線程中都調(diào)用aMethod時會產(chǎn)生同步;
2、需要注意的是這個對象所屬的類的另一對象P2卻能夠任意調(diào)用這個aMethod,因為不同的對象實例的synchronized方法是不相干擾的。也就是說,其它線程照樣可以同時訪問相同類的另一個對象實例中的aMethod方法;
3、如果一個對象有多個synchronized方法,比如aMethod、bMethod、cMethod,現(xiàn)在只要一個線程訪問了其中的一個synchronized方法,其它線程不能同時訪問這個對象中任何一個synchronized方法。
上述代碼其實等價于下面:
public void aMethod() {
synchronized (this) {
}
}
這里的this就是指的該實例對象的引用,如P1??梢娡椒椒▽嵸|(zhì)是將synchronized作用于object reference。那個拿到了P1對象鎖的線程,才能夠調(diào)用P1的同步方法,而對P2而言,P1這個鎖和他毫不相干,程序也可能在這種情形下擺脫同步機制的控制,造成數(shù)據(jù)混亂。由此我們引出了下面的同步塊。
synchronized修飾代碼塊
public void dMethod(SomeObject so) {
synchronized(so) {
}
}
在這里synchronized獲得鎖就是so這個對象的鎖,因而誰拿到這個鎖誰就能夠運行他所控制的那段代碼。當(dāng)有一個明確的對象作為鎖時,就能夠這樣寫程式,但當(dāng)沒有明確的對象作為鎖,只是想讓一段代碼同步時,能夠創(chuàng)建一個特別的instance變量(他得是個對象)來充當(dāng)鎖:
class Foo implements Runnable {
private byte[] lock = new byte[0];
Public void method() {
synchronized(lock) {
}
}
}
零長度的byte數(shù)組對象創(chuàng)建起來將比任何對象都經(jīng)濟高效。
synchronized修飾靜態(tài)方法
前面提到了synchronized關(guān)鍵字只對不同線程中的P1實例有效,那如何可以同時對P1和P2不同實例有效呢,答案就是使用synchronized修飾靜態(tài)方法,類的靜態(tài)方法可以說是這個類自有的,并不依賴類的實例,所以我們只要對類的靜態(tài)方法使用synchronized關(guān)鍵字來修飾就可以達(dá)到不同實例間的同步了。
相關(guān)文章
并發(fā)編程之Java內(nèi)存模型volatile的內(nèi)存語義
這篇文章主要介紹了并發(fā)編程之Java內(nèi)存模型volatile的內(nèi)存語義,理解volatile特性的一個好辦法是把對volatile變量的單個讀/寫,看成是使用同一個鎖對單個讀/寫操作做了同步。下面我們一起進入文章看看具體例子吧,需要的小伙伴可以參考下2021-11-11mybatis-plus數(shù)據(jù)權(quán)限實現(xiàn)代碼
這篇文章主要介紹了mybatis-plus數(shù)據(jù)權(quán)限實現(xiàn),結(jié)合了mybatis-plus的插件方式,做出了自己的注解方式的數(shù)據(jù)權(quán)限,雖然可能存在一部分的局限性,但很好的解決了我們自己去解析SQL的功能,需要的朋友可以參考下2023-06-06Springboot整合minio實現(xiàn)文件服務(wù)的教程詳解
這篇文章主要介紹了Springboot整合minio實現(xiàn)文件服務(wù)的教程,文中的示例代碼講解詳細(xì),對我們的工作或?qū)W習(xí)有一定幫助,需要的可以參考一下2022-06-06