這篇文章主要介紹“Java多線程中的虛假喚醒和怎么避免”,在日常操作中,相信很多人在Java多線程中的虛假喚醒和怎么避免問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”Java多線程中的虛假喚醒和怎么避免”的疑惑有所幫助!接下來,請跟著小編一起來學習吧!
創(chuàng)新互聯(lián)建站是專業(yè)的閻良網(wǎng)站建設(shè)公司,閻良接單;提供做網(wǎng)站、成都網(wǎng)站制作,網(wǎng)頁設(shè)計,網(wǎng)站設(shè)計,建網(wǎng)站,PHP網(wǎng)站建設(shè)等專業(yè)做網(wǎng)站服務(wù);采用PHP框架,可快速的進行閻良網(wǎng)站開發(fā)網(wǎng)頁制作和功能擴展;專業(yè)做搜索引擎喜愛的網(wǎng)站,專業(yè)的做網(wǎng)站團隊,希望更多企業(yè)前來合作!
一個賣面的面館,有一個做面的廚師和一個吃面的食客,需要保證,廚師做一碗面,食客吃一碗面,不能一次性多做幾碗面,更不能沒有面的時候吃面;按照上述操作,進行十輪做面吃面的操作。
首先我們需要有一個資源類,里面包含面的數(shù)量,做面操作,吃面操作;
當面的數(shù)量為0時,廚師才做面,做完面,需要喚醒等待的食客,否則廚師需要等待食客吃完面才能做面;
當面的數(shù)量不為0時,食客才能吃面,吃完面需要喚醒正在等待的廚師,否則食客需要等待廚師做完面才能吃面;
然后在主類中,我們創(chuàng)建一個廚師線程進行10次做面,一個食客線程進行10次吃面;
代碼如下:
package com.duoxiancheng.code; /** * @user: code隨筆 */ class Noodles{ //面的數(shù)量 private int num = 0; //做面方法 public synchronized void makeNoodles() throws InterruptedException { //如果面的數(shù)量不為0,則等待食客吃完面再做面 if(num != 0){ this.wait(); } num++; System.out.println(Thread.currentThread().getName()+"做好了一份面,當前有"+num+"份面"); //面做好后,喚醒食客來吃 this.notifyAll(); } //吃面方法 public synchronized void eatNoodles() throws InterruptedException { //如果面的數(shù)量為0,則等待廚師做完面再吃面 if(num == 0){ this.wait(); } num--; System.out.println(Thread.currentThread().getName()+"吃了一份面,當前有"+num+"份面"); //吃完則喚醒廚師來做面 this.notifyAll(); } } public class Test { public static void main(String[] args) { Noodles noodles = new Noodles(); new Thread(new Runnable(){ @Override public void run() { try { for (int i = 0; i < 10 ; i++) { noodles.makeNoodles(); } } catch (InterruptedException e) { e.printStackTrace(); } } },"廚師A").start(); new Thread(new Runnable(){ @Override public void run() { try { for (int i = 0; i < 10 ; i++) { noodles.eatNoodles(); } } catch (InterruptedException e) { e.printStackTrace(); } } },"食客甲").start(); } }
輸出如下:
可以見到是交替輸出的;
Noodles類的代碼不用動,在主類中多創(chuàng)建兩個線程即可,主類代碼如下:
public class Test { public static void main(String[] args) { Noodles noodles = new Noodles(); new Thread(new Runnable(){ @Override public void run() { try { for (int i = 0; i < 10 ; i++) { noodles.makeNoodles(); } } catch (InterruptedException e) { e.printStackTrace(); } } },"廚師A").start(); new Thread(new Runnable(){ @Override public void run() { try { for (int i = 0; i < 10 ; i++) { noodles.makeNoodles(); } } catch (InterruptedException e) { e.printStackTrace(); } } },"廚師B").start(); new Thread(new Runnable(){ @Override public void run() { try { for (int i = 0; i < 10 ; i++) { noodles.eatNoodles(); } } catch (InterruptedException e) { e.printStackTrace(); } } },"食客甲").start(); new Thread(new Runnable(){ @Override public void run() { try { for (int i = 0; i < 10 ; i++) { noodles.eatNoodles(); } } catch (InterruptedException e) { e.printStackTrace(); } } },"食客乙").start(); } }
此時輸出如下:
上面的問題就是"虛假喚醒"。
當我們只有一個廚師一個食客時,只能是廚師做面或者食客吃面,并沒有其他情況;
但是當有兩個廚師,兩個食客時,就會出現(xiàn)下面的問題:
初始狀態(tài)
廚師A得到操作權(quán),發(fā)現(xiàn)面的數(shù)量為0,可以做面,面的份數(shù)+1,然后喚醒所有線程;
廚師B得到操作權(quán),發(fā)現(xiàn)面的數(shù)量為1,不可以做面,執(zhí)行wait操作;
廚師A得到操作權(quán),發(fā)現(xiàn)面的數(shù)量為1,不可以做面,執(zhí)行wait操作;
食客甲得到操作權(quán),發(fā)現(xiàn)面的數(shù)量為1,可以吃面,吃完面后面的數(shù)量-1,并喚醒所有線程;
6. 此時廚師A得到操作權(quán)了,因為是從剛才阻塞的地方繼續(xù)運行,就不用再判斷面的數(shù)量是否為0了,所以直接面的數(shù)量+1,并喚醒其他線程;
7. 此時廚師B得到操作權(quán)了,因為是從剛才阻塞的地方繼續(xù)運行,就不用再判斷面的數(shù)量是否為0了,所以直接面的數(shù)量+1,并喚醒其他線程; 這便是虛假喚醒,還有其他的情況,讀者可以嘗試畫畫圖分析分析。
出現(xiàn)虛假喚醒的原因是從阻塞態(tài)到就緒態(tài)再到運行態(tài)沒有進行判斷,我們只需要讓其每次得到操作權(quán)時都進行判斷就可以了;
所以將
if(num != 0){ this.wait(); }
改為
while(num != 0){ this.wait(); }
將
if(num == 0){ this.wait(); }
改為
while(num == 0){ this.wait(); }
到此,關(guān)于“Java多線程中的虛假喚醒和怎么避免”的學習就結(jié)束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學習,快去試試吧!若想繼續(xù)學習更多相關(guān)知識,請繼續(xù)關(guān)注創(chuàng)新互聯(lián)網(wǎng)站,小編會繼續(xù)努力為大家?guī)砀鄬嵱玫奈恼拢?/p>
文章名稱:Java多線程中的虛假喚醒和怎么避免
文章網(wǎng)址:http://jinyejixie.com/article36/ggshsg.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供小程序開發(fā)、標簽優(yōu)化、移動網(wǎng)站建設(shè)、網(wǎng)站維護、軟件開發(fā)、定制開發(fā)
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)