JAVA線程虛假喚醒

来源:https://www.cnblogs.com/un1que/archive/2020/07/04/13236905.html
-Advertisement-
Play Games

JAVA線程虛假喚醒 線程虛假喚醒問題描述 ​ 在JDK API文檔中,關於Object類的wait()方法有這樣一句話描述"線程也可以喚醒,而不會被通知,中斷或超時,即所謂的虛假喚醒 。 雖然這在實踐中很少會發生,但應用程式必須通過測試應該使線程被喚醒的條件來防範,並且如果條件不滿足則繼續等待", ...


線程虛假喚醒問題描述

​ 在JDK API文檔中,關於Object類的wait()方法有這樣一句話描述"線程也可以喚醒,而不會被通知,中斷或超時,即所謂的虛假喚醒 。 雖然這在實踐中很少會發生,但應用程式必須通過測試應該使線程被喚醒的條件來防範,並且如果條件不滿足則繼續等待",如下圖所示:

​ 在多線程的情況下,當多個線程執行了wait()方法後,需要其它線程執行notify()或者notifyAll()方法去喚醒,假如被阻塞的多個線程都被喚醒,但實際情況是被喚醒的線程中有一部分線程是不應該被喚醒的,那麼對於這些不應該被喚醒的線程而言就是虛假喚醒。

問題復現

生產者與消費者問題

​ 假設當前有4個線程分別為A,B,C,D,其中A,B線程是生產者,C,D線程是消費者,當A和B線程生產了一個數據後就去通知消費者去消費,C和D消費掉這一個數據後就通知生產者去生產,數據的大小為1。也就是說正常情況下,數據只會有0和1兩種狀態,0表示生產者該生產數據了,1表示消費者該消費數據了。

package producer_consumer;

public class PVTest {
    public static void main(String[] args) {
        Data data = new Data();
        //生產者線程A
        new Thread(() -> {
            for (int i = 0;i < 5;i++) {
                try {
                    data.increment();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        },"A").start();

        //生產者線程B
        new Thread(() -> {
            for (int i = 0;i < 5;i++) {
                try {
                    data.increment();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        },"B").start();

        //消費者線程C
        new Thread(() -> {
            for (int i = 0;i < 5;i++) {
                try {
                    data.decrement();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        },"C").start();

        //消費者線程D
        new Thread(() -> {
            for (int i = 0;i < 5;i++) {
                try {
                    data.decrement();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        },"D").start();
    }
}

//數據類
class Data {
    //表示數據個數
    private int number = 0;
    public synchronized void increment() throws InterruptedException {
        //關鍵點,這裡應該使用while迴圈
        if (number != 0) {
            this.wait();
        }
        number++;
        System.out.println(Thread.currentThread().getName()+"生產了數據:"+number);
        this.notifyAll();
    }

    public synchronized void decrement() throws InterruptedException {
        //關鍵點,這裡應該使用while迴圈
        if (number == 0) {
            this.wait();
        }
        number--;
        System.out.println(Thread.currentThread().getName()+"消費了數據:"+number);
        this.notifyAll();
    }
}

程式結果

結果分析

​ 可以看到上述結果出現了data為2的情況,不符合之前的預期,出現問題的場景是這樣的:當data為1的時候,線程A和B先後獲取鎖去生產數據的時候會被阻塞住,然後消費者C或者D消費掉數據後去notifyAll()喚醒了線程A和B,被喚醒的A和B沒有再次去判斷data狀態,就去執行後續增加數據的邏輯了,導致兩個生產者都執行了increment(),最終data出現了2這種情況。也就是說線程A和B有一個是不應該被喚醒的卻被喚醒了,出現這個問題的關鍵點在於程式中使用到了if判斷,只判斷了一次data的狀態,應該使用while迴圈去判斷

虛假喚醒問題解決

​ 正如JDK API文檔中所說在寫程式時候應該用while去替代if,上述生產者和消費者代碼中,將Data類中的increment()和decrement()方法中的if換為while即可避免線程虛假喚醒的問題。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • import pandas from docx import Document excel=pandas.read_excel(r'F:\word練習\數據.xlsx',header=None) 文件=Document(r'F:\word練習\a.docx') 表=文件.add_table(4,4) ...
  • from docx import Document from docx import WD_PARAGRAPH_ALIGNMENT w=Documeent(r'F:\word練習\a.docx') #第一種方法 t=w.add_table(3,3) t.cell(0,0).text='李先生' #第 ...
  • 線程與進程相似,但線程是一個比進程更小的執行單位。一個進程在其執行的過程中可以產生多個線程。與進程不同的是同類的多個線程共用同一塊記憶體空間和一組系統資源,所以系統在產生一個線程,或是在各個線程之間作切換工作時,負擔要比進程小得多,也正因為如此,線程也被稱為輕量級進程。 程式是含有指令和數據的文件,被 ...
  • from docx import Document w=Document(r'F:\word練習\表格.docx') #刪除表 print(len(w.tables)) t=w.tables[0] t._element.getparent().remove(t._element) print(len ...
  • from docx import Document w=Document(r'F:\word練習\表格.docx') table_1=w.tables[0] #刪除行 print(len(table_1.rows)) row2=table_1.rows[1] row2._element.getpar ...
  • 值傳遞和引用傳遞: 值傳遞和引用傳遞的區別並不是傳遞的內容。而是實參到底有沒有被覆制一份給形參。在判斷實參內容有沒有受影響的時候,要看傳的的是什麼,如果你傳遞的是個地址,那麼就看這個地址的變化會不會有影響,而不是看地址指向的對象的變化。 Java中當傳遞的參數是對象時,其實還是值傳遞的,只不過對於對 ...
  • pygame 的聲音播放 1. sound 對象 在初始化聲音設備後就可以讀取一個音樂文件到一個 Sound 對象中。pygame.mixer.sound() 接收一個文件名,也可以是一個文件對象,不過這個文件對象必須是 WAV 或者 OGG 文件。 hello_sound = pygame.mix ...
  • 在使用dubbo時,通常會遇到timeout這個屬性,timeout屬性的作用是:給某個服務調用設置超時時間,如果服務在設置的時間內未返回結果,則會拋出調用超時異常:TimeoutException,在使用的過程中,我們有時會對provider和consumer兩個配置都會設置timeout值,那麼 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...