亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Java多線程中的虛假喚醒和怎么避免

發布時間:2021-10-23 10:12:56 來源:億速云 閱讀:153 作者:iii 欄目:編程語言

這篇文章主要介紹“Java多線程中的虛假喚醒和怎么避免”,在日常操作中,相信很多人在Java多線程中的虛假喚醒和怎么避免問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”Java多線程中的虛假喚醒和怎么避免”的疑惑有所幫助!接下來,請跟著小編一起來學習吧!

先來看一個例子

一個賣面的面館,有一個做面的廚師和一個吃面的食客,需要保證,廚師做一碗面,食客吃一碗面,不能一次性多做幾碗面,更不能沒有面的時候吃面;按照上述操作,進行十輪做面吃面的操作。

用代碼說話

首先我們需要有一個資源類,里面包含面的數量,做面操作,吃面操作;
當面的數量為0時,廚師才做面,做完面,需要喚醒等待的食客,否則廚師需要等待食客吃完面才能做面;
當面的數量不為0時,食客才能吃面,吃完面需要喚醒正在等待的廚師,否則食客需要等待廚師做完面才能吃面;
然后在主類中,我們創建一個廚師線程進行10次做面,一個食客線程進行10次吃面;
代碼如下:

package com.duoxiancheng.code;

/**
 * @user: code隨筆
 */

class Noodles{

    //面的數量
    private int num = 0;

    //做面方法
    public synchronized void makeNoodles() throws InterruptedException {
        //如果面的數量不為0,則等待食客吃完面再做面
        if(num != 0){
            this.wait();
        }

        num++;
        System.out.println(Thread.currentThread().getName()+"做好了一份面,當前有"+num+"份面");
        //面做好后,喚醒食客來吃
        this.notifyAll();
    }

    //吃面方法
    public synchronized void eatNoodles() throws InterruptedException {
        //如果面的數量為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();

    }

}

輸出如下:
Java多線程中的虛假喚醒和怎么避免 可以見到是交替輸出的;

如果有兩個廚師,兩個食客,都進行10次循環呢?

Noodles類的代碼不用動,在主類中多創建兩個線程即可,主類代碼如下:

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();

    }
}

此時輸出如下:
Java多線程中的虛假喚醒和怎么避免

虛假喚醒

上面的問題就是"虛假喚醒"。
當我們只有一個廚師一個食客時,只能是廚師做面或者食客吃面,并沒有其他情況;
但是當有兩個廚師,兩個食客時,就會出現下面的問題:

  1. 初始狀態 Java多線程中的虛假喚醒和怎么避免

  2. 廚師A得到操作權,發現面的數量為0,可以做面,面的份數+1,然后喚醒所有線程;
    Java多線程中的虛假喚醒和怎么避免

  3. 廚師B得到操作權,發現面的數量為1,不可以做面,執行wait操作; Java多線程中的虛假喚醒和怎么避免

  4. 廚師A得到操作權,發現面的數量為1,不可以做面,執行wait操作; Java多線程中的虛假喚醒和怎么避免

  5. 食客甲得到操作權,發現面的數量為1,可以吃面,吃完面后面的數量-1,并喚醒所有線程;

Java多線程中的虛假喚醒和怎么避免 6. 此時廚師A得到操作權了,因為是從剛才阻塞的地方繼續運行,就不用再判斷面的數量是否為0了,所以直接面的數量+1,并喚醒其他線程;

Java多線程中的虛假喚醒和怎么避免 7. 此時廚師B得到操作權了,因為是從剛才阻塞的地方繼續運行,就不用再判斷面的數量是否為0了,所以直接面的數量+1,并喚醒其他線程; Java多線程中的虛假喚醒和怎么避免 這便是虛假喚醒,還有其他的情況,讀者可以嘗試畫畫圖分析分析。

解決方法

出現虛假喚醒的原因是從阻塞態到就緒態再到運行態沒有進行判斷,我們只需要讓其每次得到操作權時都進行判斷就可以了;
所以將

if(num != 0){
	this.wait();
}

改為

while(num != 0){
	this.wait();
}

if(num == 0){
	this.wait();
}

改為

while(num == 0){
	this.wait();
}

到此,關于“Java多線程中的虛假喚醒和怎么避免”的學習就結束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學習,快去試試吧!若想繼續學習更多相關知識,請繼續關注億速云網站,小編會繼續努力為大家帶來更多實用的文章!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

咸阳市| 曲周县| 资阳市| 武穴市| 贵南县| 临清市| 白河县| 西充县| 金山区| 深圳市| 横山县| 彩票| 孝感市| 玉林市| 中山市| 博湖县| 清徐县| 堆龙德庆县| 高碑店市| 通州市| 神农架林区| 南郑县| 龙陵县| 云南省| 辛集市| 沾益县| 沙洋县| 通榆县| 汤原县| 桦川县| 宁南县| 富阳市| 左权县| 乌什县| 东平县| 博湖县| 宁化县| 塔城市| 桐城市| 云阳县| 武山县|