您好,登錄后才能下訂單哦!
這篇文章運用簡單易懂的例子給大家介紹synchronized如何在Java項目中使用,內容非常詳細,感興趣的小伙伴們可以參考借鑒,希望對大家能有所幫助。
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實現加鎖,同步代碼塊則在方法內部加鎖,很明顯,同步方法鎖的范圍比較大,而同步代碼塊范圍要小點,一般同步的范圍越大,性能就越差,一般需要加鎖進行同步的時候,肯定是范圍越小越好,這樣性能更好*。
關于synchronized如何在Java項目中使用就分享到這里了,希望以上內容可以對大家有一定的幫助,可以學到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。