Java 同步鎖(synchronized)詳解及實(shí)例
Java 同步鎖(synchronized)詳解及實(shí)例
Java中cpu分給每個(gè)線程的時(shí)間片是隨機(jī)的并且在Java中好多都是多個(gè)線程共用一個(gè)資源,比如火車(chē)賣(mài)票,火車(chē)票是一定的,但賣(mài)火車(chē)票的窗口到處都有,每個(gè)窗口就相當(dāng)于一個(gè)線程,這么多的線程共用所有的火車(chē)票這個(gè)資源。如果在一個(gè)時(shí)間點(diǎn)上,兩個(gè)線程同時(shí)使用這個(gè)資源,那他們?nèi)〕龅幕疖?chē)票是一樣的(座位號(hào)一樣),這樣就會(huì)給乘客造成麻煩。比如下面程序:
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()+"號(hào)窗口賣(mài)出:"+this.ticket--+"號(hào)票"); } } } public static void main(String args[]){ Runnable_demo demo=new Runnable_demo(); //基于火車(chē)票創(chuàng)建三個(gè)窗口 new Thread(demo,"a").start(); new Thread(demo,"b").start(); new Thread(demo,"c").start(); } }
程序運(yùn)行結(jié)果:
我們可以看到c號(hào)窗口和和b號(hào)窗口都賣(mài)出了10號(hào)票,并且a號(hào)和b號(hào)窗口分別賣(mài)出了0號(hào)和-1號(hào)票。造成這種情況的原因是1、c線程和b線程在ticket=10的時(shí)候,c線程取出10號(hào)票以后,ticket還沒(méi)來(lái)的及減1,b線程就取出了ticket此時(shí)ticket還等于10;2、在ticket=1時(shí),c線程取出了1號(hào)票,ticket還沒(méi)來(lái)的及減1,a、b線程就先后進(jìn)入了if判斷語(yǔ)句,這時(shí)ticket減1了,那么當(dāng)a、b線程取票的時(shí)候就取到了0號(hào)和-1號(hào)票。
出現(xiàn)了上述情況怎樣改變呢,我們可以這樣做:當(dāng)一個(gè)線程要使用火車(chē)票這個(gè)資源時(shí),我們就交給它一把鎖,等它把事情做完后在把鎖給另一個(gè)要用這個(gè)資源的線程。這樣就不會(huì)出現(xiàn)上述情況。 實(shí)現(xiàn)這個(gè)鎖的功能就需要用到synchronized這個(gè)關(guān)鍵字。
synchronized這個(gè)關(guān)鍵字有兩種用法1、放方法名前形成同步方法;2、放在塊前構(gòu)成同步塊。
1、使用同步方法將上面的例子該為:
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()+"號(hào)窗口賣(mài)出:"+this.ticket--+"號(hào)票"); } } public static void main(String args[]){ Runnable_demo demo=new Runnable_demo(); //基于火車(chē)票創(chuàng)建三個(gè)窗口 new Thread(demo,"a").start(); new Thread(demo,"b").start(); new Thread(demo,"c").start(); } }
程序的輸出結(jié)果為:
2、使用同步塊修改上面的例子:
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()+"號(hào)窗口賣(mài)出:"+this.ticket--+"號(hào)票"); } } } } public static void main(String args[]){ Runnable_demo demo=new Runnable_demo(); //基于火車(chē)票創(chuàng)建三個(gè)窗口 new Thread(demo,"a").start(); new Thread(demo,"b").start(); new Thread(demo,"c").start(); } }
程序的輸出結(jié)果:
感謝閱讀,希望能幫助到大家,謝謝大家對(duì)本站的支持!
相關(guān)文章
springboot-jpa的實(shí)現(xiàn)操作
這篇文章主要介紹了springboot-jpa的實(shí)現(xiàn)操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2021-03-03使用Spring Boot Mybatis 搞反向工程的步驟
這篇文章主要介紹了使用Spring Boot Mybatis 搞反向工程的步驟,幫助大家更好的理解和使用spring boot框架,感興趣的朋友可以了解下2021-01-01利用Kotlin + Spring Boot實(shí)現(xiàn)后端開(kāi)發(fā)
這篇文章主要給大家介紹了關(guān)于利用Kotlin + Spring Boot實(shí)現(xiàn)后端開(kāi)發(fā)的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2018-11-11