java中的synchronized同步程式碼塊和同步方法的區別
問題的由來:
看到這樣一個面試題:
//下列兩個方法有什麼區別
public synchronized void method1(){}
public void method2(){
synchronized (obj){}
}
synchronized用於解決同步問題,當有多條執行緒同時訪問共享資料時,如果不進行同步,就會發生錯誤,java提供的解決方案是:只要將操作共享資料的語句在某一時段讓一個執行緒執行完,在執行過程中,其他執行緒不能進來執行可以。解決這個問題。這裡在用synchronized時會有兩種方式,一種是上面的同步方法,即用synchronized來修飾方法,另一種是提供的同步程式碼塊。
這裡總感覺怪怪的,這兩種方法有什麼區別呢?讓我們看下程式碼:
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(); } }
這段小程式碼片段列印結果如下:
methodA.....
methodC.....
//methodB會隔一段時間才會打印出來
methodB.....
這段程式碼的列印結果是,methodA…..methodC…..會很快打印出來,methodB…..會隔一段時間才打印出來,那麼methodB為什麼不能像methodC那樣很快被呼叫呢?
在啟動執行緒1呼叫方法A後,接著會讓執行緒1休眠5秒鐘,這時會呼叫方法C,注意到方法C這裡用synchronized進行加鎖,這裡鎖的物件是str這個字串物件。但是方法B則不同,是用當前物件this進行加鎖,注意到方法A直接在方法上加synchronized,這個加鎖的物件是什麼呢?顯然,這兩個方法用的是一把鎖。
*由這樣的結果,我們就知道這樣同步方法是用什麼加鎖的了,由於執行緒1在休眠,這時鎖還沒釋放,導致執行緒2只有在5秒之後才能呼叫方法B,由此,可知兩種加鎖機制用的是同一個鎖物件,即當前物件。
另外,同步方法直接在方法上加synchronized實現加鎖,同步程式碼塊則在方法內部加鎖,很明顯,同步方法鎖的範圍比較大,而同步程式碼塊範圍要小點,一般同步的範圍越大,效能就越差,一般需要加鎖進行同步的時候,肯定是範圍越小越好,這樣效能更好*。
同步程式碼塊可以用更細粒度的控制鎖,比如:
public class Test{
private String name = "xiaoming";
private String id = "0753";
public void setName(String name) {
synchornized(name) {
this.name = name;
}
}
public void setId(String id) {
synchornized(id) {
this.id = id;
}
}
}
如果你有一個Test物件 你想在多執行緒下同時修改Name和id, 如果你兩個set方法都宣告為同步方法,那麼在同一時間只能修改name或者id. 但是這兩個是可以同時修改的,所以你需要同步程式碼塊,將訊號量分別設定成name和id.