国产成人精品久久免费动漫-国产成人精品天堂-国产成人精品区在线观看-国产成人精品日本-a级毛片无码免费真人-a级毛片毛片免费观看久潮喷

您的位置:首頁技術文章
文章詳情頁

Java wait和notify虛假喚醒原理

瀏覽:4日期:2022-09-02 17:50:07

自己在此記錄一下,方便日后復習。

虛假喚醒的概念

jdk官方文檔解釋:

Java wait和notify虛假喚醒原理

所以說在wait和notify一塊使用時,如果使用if作為條件時,會有虛假喚醒的情況發生,所以必須使用while作為循環條件。下面來舉例實驗:

首先,創建一個資源類:(在多線程中,一般都是資源類和線程操作解耦,不放在用同一個類中,只有在線程操作資源類時,才會創建資源類的對象)

package com.test;/** * 資源類 * @author Huxudong * @createTime 2020-04-01 21:57:39 **/public class Resource { /** 產品數 */ private int product = 0; /** 進貨 */ public synchronized void get() { if(product >= 10) { System.out.println(Thread.currentThread().getName()+':'+'產品已滿!'); /** 當商品已經滿的時候,進貨線程掛起 */ try {this.wait(); } catch (InterruptedException e) {e.printStackTrace(); } } /** 進貨 */ System.out.println(Thread.currentThread().getName()+':'+ ++product); /** 喚醒其他線程 */ this.notifyAll(); } /** 售貨 */ public synchronized void sale() { if(product <= 0) { System.out.println(Thread.currentThread().getName()+':'+'產品已空'); try {this.wait(); } catch (InterruptedException e) {e.printStackTrace(); } } /** 售貨 */ System.out.println(Thread.currentThread().getName()+':'+ --product); /** 喚醒其他線程 */ this.notify(); }}

然后再創建線程來操作我們的資源類(通過java8新特性Lambda表達式直接創建)

package com.test;import java.util.concurrent.TimeUnit;/** * 線程操作資源類,實現線程與資源類的解耦合 * @author Huxudong * @createTime 2020-04-01 23:13:54 **/public class TestPc { public static void main(String[] args) { Resource resource = new Resource(); new Thread(()->{ for (int i = 0; i < 20; i++) {try { /** 睡眠,便于觀察結果 */ TimeUnit.SECONDS.sleep(2);} catch (InterruptedException e) { e.printStackTrace();}resource.get(); } },'生產者A').start(); new Thread(()->{ for (int i = 0; i < 20; i++) {resource.sale(); } },'消費者C').start(); new Thread(()->{ for (int i = 0; i < 20; i++) {resource.get(); } },'生產者B').start(); new Thread(()->{ for (int i = 0; i < 20; i++) {resource.sale(); } },'消費者D').start(); }}

先來看看如果使用if條件會發生什么:

Java wait和notify虛假喚醒原理

對,你沒看錯,怎么可能會出現負數呢,這肯定是不對的。冷靜下來分析一下,還是有點頭緒,知道哪里出現了問題的(那你是一個處事不驚的人,很厲害)。

來,分析一下,一開始先調用了消費者C,D線程(因為我們寫了睡眠在生產者中),消費者此時發現此時product資源為0,所以,消費者C,D這兩個兄弟,沒辦法只能調用wait方法,睡眠了,并且釋放了鎖。

但是此時第一個消費者已經蘇醒了,發動機開始生產產品了,并且生產之后,又喚醒了所有等待的消費者線程。這消費者C,D兩兄弟終于蘇醒了,D哥們先獲得了鎖,所以就先消費了一個產品,然后就又發現沒有產品了,又傷心的休眠去了,但是不要忘了,此時還有一個C哥們被喚醒了啊,你喚醒了人家,人家總的干點什么事情吧,不然這多難受,剛好不巧的是,此時的判斷條件是if,所以此時C哥們便不受條件的約束,接著上面自己睡眠的代碼處執行,毅然決然的又去消費了一個產品,原來D哥們消費后,就已經為0了,這個C哥們再去消費減一,不就是-1了嗎,以此類推分析。發現如果判斷條件用不好,此時喚醒的C哥們就相當于虛假喚醒的了,會給程序帶來不可預估的錯誤。所以在這里判斷必須要使用while,先來看看把if換成while的結果。

Java wait和notify虛假喚醒原理

這回結果就比較正常了,為什么使用while就可以呢,因為像上文所說,即使喚醒了所有的消費者線程,此時會不停while循環判斷,如果此時條件是為0,那么C哥們就不能出while,那么他也就不回執行下面消費產品的減減操作了,那么就會避免了這種錯誤。這也是官方提倡的在使用wait 和notifyAll的時候,必須使用while循環條件判斷。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持好吧啦網。

標簽: Java
相關文章:
主站蜘蛛池模板: 久久精品91 | 美女扒开腿让男生桶爽网站 | 中文字幕免费视频 | 好看的看黄a大片爽爽影院 好男人天堂网 | 岛国精品成人 | 上海一级毛片 | 成年男女男精品免费视频网站 | 久久久久久久久久久观看 | 在线观看免费视频国产 | 毛片96视频免费观看 | 欧美一级二级三级视频 | 精品乱人伦一区二区 | 中文字幕日本一区波多野不卡 | 世界一级毛片 | 久久a 热6| 成人性版蝴蝶影院污 | 色偷偷亚洲男人天堂 | 中文无码日韩欧免费视频 | 毛片看 | 毛片韩国| 国产亚洲精品yxsp | 久久福利青草精品资源 | 国内精品久久久久不卡 | 精品中文字幕一区在线 | 亚洲精品一区二区三区网址 | 亚洲精品视频观看 | 曰本一区 | 2017天天爽夜夜爽精品视频 | 亚洲欧洲日产国码二区首页 | 欧美一区二区三区免费 | 日韩精品福利视频一区二区三区 | 亚洲视频中文字幕在线观看 | 成人做爰全过程免费看网站 | 国产国语高清在线视频二区 | 国产三级日本三级在线播放 | 免费观看国产精品 | 欧毛片 | 黄在线观看网站 | 欧美色欧美亚洲高清在线视频 | 成人免费在线视频网 | 免费毛片全部不收费的 |