Java 同步鎖(synchronized)詳解及實例
Java中cpu分給每個線程的時間片是隨機的并且在Java中好多都是多個線程共用一個資源,比如火車賣票,火車票是一定的,但賣火車票的窗口到處都有,每個窗口就相當于一個線程,這么多的線程共用所有的火車票這個資源。如果在一個時間點上,兩個線程同時使用這個資源,那他們取出的火車票是一樣的(座位號一樣),這樣就會給乘客造成麻煩。比如下面程序:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
|
package com.pakage.ThreadAndRunnable; public class Runnable_demo implements Runnable{ private int ticket= 10 ; public Runnable_demo(){ } @Override public void run() { for ( int i= 0 ;i< 20 ;i++){ if ( this .ticket> 0 ){ //休眠1s秒中,為了使效果更明顯,否則可能出不了效果 try { Thread.sleep( 1000 ); } catch (Exception e) { e.printStackTrace(); } System.out.println(Thread.currentThread().getName()+ "號窗口賣出:" + this .ticket--+ "號票" ); } } } public static void main(String args[]){ Runnable_demo demo= new Runnable_demo(); //基于火車票創建三個窗口 new Thread(demo, "a" ).start(); new Thread(demo, "b" ).start(); new Thread(demo, "c" ).start(); } } |
程序運行結果:
我們可以看到c號窗口和和b號窗口都賣出了10號票,并且a號和b號窗口分別賣出了0號和-1號票。造成這種情況的原因是1、c線程和b線程在ticket=10的時候,c線程取出10號票以后,ticket還沒來的及減1,b線程就取出了ticket此時ticket還等于10;2、在ticket=1時,c線程取出了1號票,ticket還沒來的及減1,a、b線程就先后進入了if判斷語句,這時ticket減1了,那么當a、b線程取票的時候就取到了0號和-1號票。
出現了上述情況怎樣改變呢,我們可以這樣做:當一個線程要使用火車票這個資源時,我們就交給它一把鎖,等它把事情做完后在把鎖給另一個要用這個資源的線程。這樣就不會出現上述情況。 實現這個鎖的功能就需要用到synchronized這個關鍵字。
synchronized這個關鍵字有兩種用法1、放方法名前形成同步方法;2、放在塊前構成同步塊。
1、使用同步方法將上面的例子該為:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
|
package com.pakage.ThreadAndRunnable; public class Runnable_demo implements Runnable{ private int ticket= 10 ; public Runnable_demo(){ } @Override public void run() { for ( int i= 0 ;i< 20 ;i++){ if ( this .ticket> 0 ){ //休眠1s秒中,為了使效果更明顯,否則可能出不了效果 try { Thread.sleep( 1000 ); } catch (Exception e) { e.printStackTrace(); } this .sale(); } } } public synchronized void sale(){ if ( this .ticket> 0 ){ System.out.println(Thread.currentThread().getName()+ "號窗口賣出:" + this .ticket--+ "號票" ); } } public static void main(String args[]){ Runnable_demo demo= new Runnable_demo(); //基于火車票創建三個窗口 new Thread(demo, "a" ).start(); new Thread(demo, "b" ).start(); new Thread(demo, "c" ).start(); } } |
程序的輸出結果為:
2、使用同步塊修改上面的例子:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
|
package com.pakage.ThreadAndRunnable; public class Runnable_demo implements Runnable{ private int ticket= 10 ; public Runnable_demo(){ } @Override public void run() { for ( int i= 0 ;i< 20 ;i++){ <span style= "color:#ff0000" > synchronized </span>( this ){ if ( this .ticket> 0 ){ //休眠1s秒中,為了使效果更明顯,否則可能出不了效果 try { Thread.sleep( 1000 ); } catch (Exception e) { e.printStackTrace(); } System.out.println(Thread.currentThread().getName()+ "號窗口賣出:" + this .ticket--+ "號票" ); } } } } public static void main(String args[]){ Runnable_demo demo= new Runnable_demo(); //基于火車票創建三個窗口 new Thread(demo, "a" ).start(); new Thread(demo, "b" ).start(); new Thread(demo, "c" ).start(); } } |
程序的輸出結果:
感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!
原文鏈接:http://blog.csdn.net/qq_35101189/article/details/67634112