java中synchronized(同步代碼塊和同步方法)詳解及區(qū)別
java中synchronized(同步代碼塊和同步方法)詳解及區(qū)別
問題的由來(lái):
看到這樣一個(gè)面試題:
//下列兩個(gè)方法有什么區(qū)別
public synchronized void method1(){}
public void method2(){
synchronized (obj){}
}
synchronized用于解決同步問題,當(dāng)有多條線程同時(shí)訪問共享數(shù)據(jù)時(shí),如果進(jìn)行同步,就會(huì)發(fā)生錯(cuò)誤,Java提供的解決方案是:只要將操作共享數(shù)據(jù)的語(yǔ)句在某一時(shí)段讓一個(gè)線程執(zhí)行完,在執(zhí)行過程中,其他線程不能進(jìn)來(lái)執(zhí)行可以。解決這個(gè)問題。這里在用synchronized時(shí)會(huì)有兩種方式,一種是上面的同步方法,即用synchronized來(lái)修飾方法,另一種是提供的同步代碼塊。
這里總感覺怪怪的,這兩種方法有什么區(qū)別呢,基礎(chǔ)學(xué)得不好,于是就動(dòng)手做了個(gè)簡(jiǎn)單的測(cè)試,代碼如下:
public class SynObj {
public synchronized void methodA() {
System.out.println("methodA.....");
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
public void methodB() {
synchronized(this) {
System.out.pritntln("methodB.....");
}
}
public void methodC() {
String str = "sss";
synchronized (str) {
System.out.println( "methodC.....");
}
}
}
public class TestSyn {
public static void main(String[] args) {
final SynObj obj = new SynObj();
Thread t1 = new Thread(new Runnable() {
@Override
public void run() {
obj.methodA();
}
});
t1.start();
Thread t2 = new Thread(new Runnable() {
@Override
public void run() {
obj.methodB();
}
});
t2.start();
Thread t3 = new Thread(new Runnable() {
@Override
public void run() {
obj.methodC();
}
});
t3.start();
}
}
這段小代碼片段打印結(jié)果如下:
methodA..... methodC..... //methodB會(huì)隔一段時(shí)間才會(huì)打印出來(lái) methodB.....
這段代碼的打印結(jié)果是,methodA…..methodC…..會(huì)很快打印出來(lái),methodB…..會(huì)隔一段時(shí)間才打印出來(lái),那么methodB為什么不能像methodC那樣很快被調(diào)用呢?
在啟動(dòng)線程1調(diào)用方法A后,接著會(huì)讓線程1休眠5秒鐘,這時(shí)會(huì)調(diào)用方法C,注意到方法C這里用synchronized進(jìn)行加鎖,這里鎖的對(duì)象是str這個(gè)字符串對(duì)象。但是方法B則不同,是用當(dāng)前對(duì)象this進(jìn)行加鎖,注意到方法A直接在方法上加synchronized,這個(gè)加鎖的對(duì)象是什么呢?顯然,這兩個(gè)方法用的是一把鎖。
*由這樣的結(jié)果,我們就知道這樣同步方法是用什么加鎖的了,由于線程1在休眠,這時(shí)鎖還沒釋放,導(dǎo)致線程2只有在5秒之后才能調(diào)用方法B,由此,可知兩種加鎖機(jī)制用的是同一個(gè)鎖對(duì)象,即當(dāng)前對(duì)象。
另外,同步方法直接在方法上加synchronized實(shí)現(xiàn)加鎖,同步代碼塊則在方法內(nèi)部加鎖,很明顯,同步方法鎖的范圍比較大,而同步代碼塊范圍要小點(diǎn),一般同步的范圍越大,性能就越差,一般需要加鎖進(jìn)行同步的時(shí)候,肯定是范圍越小越好,這樣性能更好*。
感謝閱讀,希望能幫助到大家,謝謝大家對(duì)本站的支持!
相關(guān)文章
Java分析Lambda表達(dá)式Stream流合并分組內(nèi)對(duì)象數(shù)據(jù)合并
Lambda表達(dá)式,基于Lambda所帶來(lái)的函數(shù)式編程,又引入了一個(gè)全新的Stream概念,用于解決集合類庫(kù)既有的弊端,Lambda 允許把函數(shù)作為一個(gè)方法的參數(shù)(函數(shù)作為參數(shù)傳遞進(jìn)方法中)。使用 Lambda 表達(dá)式可以使代碼變的更加簡(jiǎn)潔緊湊2022-12-12
springboot整合freemarker代碼自動(dòng)生成器
最近做了一個(gè)工具,可以實(shí)現(xiàn)代碼自動(dòng)生成,今天整理出來(lái)分享給大家,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-05-05
SpringBoot+Dubbo+Seata分布式事務(wù)實(shí)戰(zhàn)詳解
這篇文章主要介紹了SpringBoot+Dubbo+Seata分布式事務(wù)實(shí)戰(zhàn)詳解,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-07-07
Springboot項(xiàng)目平滑關(guān)閉及自動(dòng)化關(guān)閉腳本
這篇文章主要為大家詳細(xì)介紹了Springboot項(xiàng)目平滑關(guān)閉及自動(dòng)化關(guān)閉腳本,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-05-05
SpringBoot中的配置文件加載優(yōu)先級(jí)詳解
這篇文章主要介紹了SpringBoot中的配置文件加載優(yōu)先級(jí)詳解,springboot啟動(dòng)會(huì)掃描以下位置的application.properties或者application.yml文件作為Spring?boot的默認(rèn)配置文件,需要的朋友可以參考下2024-01-01
Java手動(dòng)創(chuàng)建線程池代碼實(shí)例
這篇文章主要介紹了Java手動(dòng)創(chuàng)建線程池代碼實(shí)例,FixedThreadPool或者SingleThreadPool,允許的請(qǐng)求隊(duì)列長(zhǎng)度為Integer.MAX_VALUE,可能會(huì)堆積大量的請(qǐng)求,從而導(dǎo)致OOM,需要的朋友可以參考下2023-12-12

