日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > java >内容正文

java

Java并发编程—为什么 wait() 方法需要写在 while 里,而不是 if?

發布時間:2024/4/15 java 49 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Java并发编程—为什么 wait() 方法需要写在 while 里,而不是 if? 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

原文作者:后端面試那些事兒

原文地址:再見面試官:為什么 wait() 方法需要寫在 while 里,而不是 if?

  • 問:為什么是 while 而不是 if ?

  • 問:什么時候用 notifyAll 或者 notify?


問:為什么是 while 而不是 if ?

大多數人都知道常見的使用 synchronized 代碼:

synchronized?(obj)?{while?(check?pass)?{wait();}//?do?your?business }

那么問題是為啥這里是 while 而不是 if 呢?這個問題我最開始也想了很久,按理來說已經在 synchronized 塊里面了嘛,就不需要了。這個也是我前面一直是這么認為的,直到最近看了一個 Stackoverflow 上的問題才對這個問題有了比較深入的理解。

試想我們要試想一個有界的隊列。那么常見的代碼可以是這樣:

static?class?Buf?{private?final?int?MAX?=?5;private?final?ArrayList<Integer>?list?=?new?ArrayList<>();synchronized?void?put(int?v)?throws?InterruptedException?{if?(list.size()?==?MAX)?{wait();}list.add(v);notifyAll();}synchronized?int?get()?throws?InterruptedException?{//?line?0if?(list.size()?==?0)?{??//?line?1wait();??//?line2//?line?3}int?v?=?list.remove(0);??//?line?4notifyAll();?//?line?5return?v;}synchronized?int?size()?{return?list.size();} }

注意到這里用的 if,那么我們來看看它會報什么錯呢?下面的代碼用了 1 個線程來 put,10 個線程來 get:

final?Buf?buf?=?new?Buf(); ExecutorService?es?=?Executors.newFixedThreadPool(11); for?(int?i?=?0;?i?<?1;?i++) es.execute(new?Runnable()?{@Overridepublic?void?run()?{while?(true?)?{try?{buf.put(1);Thread.sleep(20);}catch?(InterruptedException?e)?{e.printStackTrace();break;}}} }); for?(int?i?=?0;?i?<?10;?i++)?{es.execute(new?Runnable()?{@Overridepublic?void?run()?{while?(true?)?{try?{buf.get();Thread.sleep(10);}catch?(InterruptedException?e)?{e.printStackTrace();break;}}}}); }es.shutdown(); es.awaitTermination(1,?TimeUnit.DAYS);

這段代碼很快或者說一開始就會報錯:

java.lang.IndexOutOfBoundsException:?Index:?0,?Size:?0 at?java.util.ArrayList.rangeCheck(ArrayList.java:653) at?java.util.ArrayList.remove(ArrayList.java:492) at?TestWhileWaitBuf.get(TestWhileWait.java:80)atTestWhileWait2.run(TestWhileWait.java:47) at?java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142) at?java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617) at?java.lang.Thread.run(Thread.java:745)

很明顯,在 remove 的時候報錯了。那么我們來分析下:

假設現在有 A,B 兩個線程來執行 get 操作,我們假設如下的步驟發生了:

\1. A 拿到了鎖 line 0。

\2. A 發現 size==0, (line 1),然后進入等待,并釋放鎖 (line 2)。

\3. 此時 B 拿到了鎖,line0,發現 size==0,(line 1),然后進入等待,并釋放鎖 (line 2)。

\4. 這個時候有個線程 C 往里面加了個數據 1,那么 notifyAll 所有的等待的線程都被喚醒了。

\5. AB 重新獲取鎖,假設又是 A 拿到了。然后他就走到 line 3,移除了一個數據,(line4) 沒有問題。

\6. A 移除數據后想通知別人,此時 list 的大小有了變化,于是調用了 notifyAll (line5),這個時候就把 B 給喚醒了,那么 B 接著往下走。

\7. 這時候 B 就出問題了,因為其實此時的競態條件已經不滿足了 (size==0)。B 以為還可以刪除就嘗試去刪除,結果就跑了異常了。

那么 fix 很簡單,在 get 的時候加上 while 就好了:

synchronized?int?get()?throws?InterruptedException?{while?(list.size()?==?0)?{wait();}int?v?=?list.remove(0);notifyAll();return?v;}

同樣的,我們可以嘗試修改 put 的線程數和 get 的線程數來發現如果 put 里面不是 while 的話也是不行的。

我們可以用一個外部周期性任務來打印當前 list 的大小,你會發現大小并不是固定的最大5:

final?Buf?buf?=?new?Buf(); ExecutorService?es?=?Executors.newFixedThreadPool(11); ScheduledExecutorService?printer?=?Executors.newScheduledThreadPool(1); printer.scheduleAtFixedRate(new?Runnable()?{@Overridepublic?void?run()?{System.out.println(buf.size());} },?0,?1,?TimeUnit.SECONDS); for?(int?i?=?0;?i?<?10;?i++) es.execute(new?Runnable()?{@Overridepublic?void?run()?{while?(true?)?{try?{buf.put(1);Thread.sleep(200);}catch?(InterruptedException?e)?{e.printStackTrace();break;}}} }); for?(int?i?=?0;?i?<?1;?i++)?{es.execute(new?Runnable()?{@Overridepublic?void?run()?{while?(true?)?{try?{buf.get();Thread.sleep(100);}catch?(InterruptedException?e)?{e.printStackTrace();break;}}}}); }es.shutdown(); es.awaitTermination(1,?TimeUnit.DAYS);

這里我想應該說清楚了為啥必須是 while 還是 if 了。

問:什么時候用 notifyAll 或者 notify?

大多數人都會這么告訴你,當你想要通知所有人的時候就用 notifyAll,當你只想通知一個人的時候就用 notify。但是我們都知道 notify 實際上我們是沒法決定到底通知誰的(都是從等待集合里面選一個)。那這個還有什么存在的意義呢?

在上面的例子中,我們用到了 notifyAll,那么下面我們來看下用 notify 是否可以工作呢?

synchronized?void?put(int?v)?throws?InterruptedException?{if?(list.size()?==?MAX)?{wait();}list.add(v);notify();}synchronized?int?get()?throws?InterruptedException?{while?(list.size()?==?0)?{wait();}int?v?=?list.remove(0);notify();return?v;}

下面的幾點是 jvm 告訴我們的:

  • 任何時候,被喚醒的來執行的線程是不可預知。比如有 5 個線程都在一個對象上,實際上我不知道 下一個哪個線程會被執行。

  • synchronized 語義實現了有且只有一個線程可以執行同步塊里面的代碼。

  • 那么我們假設下面的場景就會導致死鎖:

    P – 生產者 調用 put。C – 消費者 調用 get。

  • P1 放了一個數字1。

  • P2 想來放,發現滿了,在wait里面等了。

  • P3 想來放,發現滿了,在 wait 里面等了。

  • C1 想來拿,C2,C3 就在 get 里面等著。

  • C1 開始執行,獲取1,然后調用 notify 然后退出。

    • 如果 C1 把 C2 喚醒了,所以P2 (其他的都得等)只能在put方法上等著。(等待獲取synchoronized (this) 這個monitor)。

    • C2 檢查 while 循環發現此時隊列是空的,所以就在 wait 里面等著。

    • C3 也比 P2 先執行,那么發現也是空的,只能等著了。

  • 這時候我們發現 P2、C2、C3 都在等著鎖,最終 P2 拿到了鎖,放一個 1,notify,然后退出。

  • P2 這個時候喚醒了P3,P3發現隊列是滿的,沒辦法,只能等它變為空。

  • 這時候沒有別的調用了,那么現在這三個線程(P3, C2,C3)就全部變成 suspend 了,也就是死鎖了。

  • 總結

    以上是生活随笔為你收集整理的Java并发编程—为什么 wait() 方法需要写在 while 里,而不是 if?的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。