成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

如何在 Java 中正確使用 wait, notify 和 notifyAll

開發(fā) 后端
在 Java 中可以用 wait、notify 和 notifyAll 來實現(xiàn)線程間的通信。。舉個例子,如果你的Java程序中有兩個線程——即生產(chǎn)者和消費者,那么生產(chǎn)者可以通知消費者,讓消費者開始消耗數(shù)據(jù),因為隊列緩 沖區(qū)中有內(nèi)容待消費(不為空)。相應(yīng)的,消費者可以通知生產(chǎn)者可以開始生成更多的數(shù)據(jù),因為當(dāng)它消耗掉某些數(shù)據(jù)后緩沖區(qū)不再為滿。

wait, notify 和 notifyAll,這些在多線程中被經(jīng)常用到的保留關(guān)鍵字,在實際開發(fā)的時候很多時候卻并沒有被大家重視。本文對這些關(guān)鍵字的使用進行了描述。

在 Java 中可以用 wait、notify 和 notifyAll 來實現(xiàn)線程間的通信。。舉個例子,如果你的Java程序中有兩個線程——即生產(chǎn)者和消費者,那么生產(chǎn)者可以通知消費者,讓消費者開始消耗數(shù)據(jù),因為隊列緩 沖區(qū)中有內(nèi)容待消費(不為空)。相應(yīng)的,消費者可以通知生產(chǎn)者可以開始生成更多的數(shù)據(jù),因為當(dāng)它消耗掉某些數(shù)據(jù)后緩沖區(qū)不再為滿。

我們可以利用wait()來讓一個線程在某些條件下暫停運行。例如,在生產(chǎn)者消費者模型中,生產(chǎn)者線程在緩沖區(qū)為滿的時候,消費者在緩沖區(qū)為空的時 候,都應(yīng)該暫停運行。如果某些線程在等待某些條件觸發(fā),那當(dāng)那些條件為真時,你可以用 notify 和 notifyAll 來通知那些等待中的線程重新開始運行。不同之處在于,notify 僅僅通知一個線程,并且我們不知道哪個線程會收到通知,然而 notifyAll 會通知所有等待中的線程。換言之,如果只有一個線程在等待一個信號燈,notify和notifyAll都會通知到這個線程。但如果多個線程在等待這個信 號燈,那么notify只會通知到其中一個,而其它線程并不會收到任何通知,而notifyAll會喚醒所有等待中的線程。

在這篇文章中你將會學(xué)到如何使用 wait、notify 和 notifyAll 來實現(xiàn)線程間的通信,從而解決生產(chǎn)者消費者問題。如果你想要更深入地學(xué)習(xí)Java中的多線程同步問題,我強烈推薦閱讀Brian Goetz所著的《Java Concurrency in Practice | Java 并發(fā)實踐》,不讀這本書你的 Java 多線程征程就不完整哦!這是我最向Java開發(fā)者推薦的書之一。

如何使用Wait

盡管關(guān)于wait和notify的概念很基礎(chǔ),它們也都是Object類的函數(shù),但用它們來寫代碼卻并不簡單。如果你在面試中讓應(yīng)聘者來手寫代碼, 用wait和notify解決生產(chǎn)者消費者問題,我?guī)缀蹩梢钥隙ㄋ麄冎械拇蠖鄶?shù)都會無所適從或者犯下一些錯誤,例如在錯誤的地方使用 synchronized 關(guān)鍵詞,沒有對正確的對象使用wait,或者沒有遵循規(guī)范的代碼方法。說實話,這個問題對于不常使用它們的程序員來說確實令人感覺比較頭疼。

***個問題就是,我們怎么在代碼里使用wait()呢?因為wait()并不是Thread類下的函數(shù),我們并不能使用 Thread.call()。事實上很多Java程序員都喜歡這么寫,因為它們習(xí)慣了使用Thread.sleep(),所以他們會試圖使用wait() 來達成相同的目的,但很快他們就會發(fā)現(xiàn)這并不能順利解決問題。正確的方法是對在多線程間共享的那個Object來使用wait。在生產(chǎn)者消費者問題中,這 個共享的Object就是那個緩沖區(qū)隊列。

第二個問題是,既然我們應(yīng)該在synchronized的函數(shù)或是對象里調(diào)用wait,那哪個對象應(yīng)該被synchronized呢?答案是,那個 你希望上鎖的對象就應(yīng)該被synchronized,即那個在多個線程間被共享的對象。在生產(chǎn)者消費者問題中,應(yīng)該被synchronized的就是那個 緩沖區(qū)隊列。(我覺得這里是英文原文有問題……本來那個句末就不應(yīng)該是問號不然不太通……)

如何在 Java 中正確使用 wait, notify 和 notifyAll – 以生產(chǎn)者消費者模型為例

永遠在循環(huán)(loop)里調(diào)用 wait 和 notify,不是在 If 語句

現(xiàn)在你知道wait應(yīng)該永遠在被synchronized的背景下和那個被多線程共享的對象上調(diào)用,下一個一定要記住的問題就是,你應(yīng)該永遠在 while循環(huán),而不是if語句中調(diào)用wait。因為線程是在某些條件下等待的——在我們的例子里,即“如果緩沖區(qū)隊列是滿的話,那么生產(chǎn)者線程應(yīng)該等 待”,你可能直覺就會寫一個if語句。但if語句存在一些微妙的小問題,導(dǎo)致即使條件沒被滿足,你的線程你也有可能被錯誤地喚醒。所以如果你不在線程被喚 醒后再次使用while循環(huán)檢查喚醒條件是否被滿足,你的程序就有可能會出錯——例如在緩沖區(qū)為滿的時候生產(chǎn)者繼續(xù)生成數(shù)據(jù),或者緩沖區(qū)為空的時候消費者 開始小號數(shù)據(jù)。所以記住,永遠在while循環(huán)而不是if語句中使用wait!我會推薦閱讀《Effective Java》,這是關(guān)于如何正確使用wait和notify的***的參考資料。

基于以上認知,下面這個是使用wait和notify函數(shù)的規(guī)范代碼模板:

 

  1. // The standard idiom for calling the wait method in Java 
  2. synchronized (sharedObject) { 
  3.     while (condition) { 
  4. sharedObject.wait(); 
  5.         // (Releases lock, and reacquires on wakeup) 
  6.     } 
  7.     // do action based upon condition e.g. take or put into queue 

就像我之前說的一樣,在while循環(huán)里使用wait的目的,是在線程被喚醒的前后都持續(xù)檢查條件是否被滿足。如果條件并未改變,wait被調(diào)用之前notify的喚醒通知就來了,那么這個線程并不能保證被喚醒,有可能會導(dǎo)致死鎖問題。

Java wait(), notify(), notifyAll() 范例

下面我們提供一個使用wait和notify的范例程序。在這個程序里,我們使用了上文所述的一些代碼規(guī)范。我們有兩個線程,分別名為 PRODUCER(生產(chǎn)者)和CONSUMER(消費者),他們分別繼承了了Producer和Consumer類,而Producer和 Consumer都繼承了Thread類。Producer和Consumer想要實現(xiàn)的代碼邏輯都在run()函數(shù)內(nèi)。Main線程開始了生產(chǎn)者和消費 者線程,并聲明了一個LinkedList作為緩沖區(qū)隊列(在Java中,LinkedList實現(xiàn)了隊列的接口)。生產(chǎn)者在***循環(huán)中持續(xù)往 LinkedList里插入隨機整數(shù)直到LinkedList滿。我們在while(queue.size == maxSize)循環(huán)語句中檢查這個條件。請注意到我們在做這個檢查條件之前已經(jīng)在隊列對象上使用了synchronized關(guān)鍵詞,因而其它線程不能在 我們檢查條件時改變這個隊列。如果隊列滿了,那么PRODUCER線程會在CONSUMER線程消耗掉隊列里的任意一個整數(shù),并用notify來通知 PRODUCER線程之前持續(xù)等待。在我們的例子中,wait和notify都是使用在同一個共享對象上的。

  1. import java.util.LinkedList; 
  2. import java.util.Queue; 
  3. import java.util.Random; 
  4. /** 
  5. * Simple Java program to demonstrate How to use wait, notify and notifyAll() 
  6. * method in Java by solving producer consumer problem. 
  7. * 
  8. * @author Javin Paul 
  9. */ 
  10. public class ProducerConsumerInJava { 
  11. public static void main(String args[]) { 
  12.   System.out.println("How to use wait and notify method in Java"); 
  13.   System.out.println("Solving Producer Consumper Problem"); 
  14.   Queue<Integer> buffer = new LinkedList<>(); 
  15.   int maxSize = 10
  16.   Thread producer = new Producer(buffer, maxSize, "PRODUCER"); 
  17.   Thread consumer = new Consumer(buffer, maxSize, "CONSUMER"); 
  18.   producer.start(); consumer.start(); } 
  19. /** 
  20. * Producer Thread will keep producing values for Consumer 
  21. * to consumer. It will use wait() method when Queue is full 
  22. * and use notify() method to send notification to Consumer 
  23. * Thread. 
  24. * 
  25. * @author WINDOWS 8 
  26. * 
  27. */ 
  28. class Producer extends Thread 
  29. private Queue<Integer> queue; 
  30.   private int maxSize; 
  31.   public Producer(Queue<Integer> queue, int maxSize, String name){ 
  32.    super(name); this.queue = queue; this.maxSize = maxSize; 
  33.   } 
  34.   @Override public void run() 
  35.   { 
  36.    while (true
  37.     { 
  38.      synchronized (queue) { 
  39.       while (queue.size() == maxSize) { 
  40.        try { 
  41.         System.out .println("Queue is full, " + "Producer thread waiting for " + "consumer to take something from queue"); 
  42.         queue.wait(); 
  43.        } catch (Exception ex) { 
  44.         ex.printStackTrace(); } 
  45.        } 
  46.        Random random = new Random(); 
  47.        int i = random.nextInt(); 
  48.        System.out.println("Producing value : " + i); queue.add(i); queue.notifyAll(); 
  49.       } 
  50.      } 
  51.     } 
  52.    } 
  53. /** 
  54. * Consumer Thread will consumer values form shared queue. 
  55. * It will also use wait() method to wait if queue is 
  56. * empty. It will also use notify method to send 
  57. * notification to producer thread after consuming values 
  58. * from queue. 
  59. * 
  60. * @author WINDOWS 8 
  61. * 
  62. */ 
  63. class Consumer extends Thread { 
  64.   private Queue<Integer> queue; 
  65.   private int maxSize; 
  66.   public Consumer(Queue<Integer> queue, int maxSize, String name){ 
  67.    super(name); 
  68.    this.queue = queue; 
  69.    this.maxSize = maxSize; 
  70.   } 
  71.   @Override public void run() { 
  72.    while (true) { 
  73.     synchronized (queue) { 
  74.      while (queue.isEmpty()) { 
  75.       System.out.println("Queue is empty," + "Consumer thread is waiting" + " for producer thread to put something in queue"); 
  76.       try { 
  77.        queue.wait(); 
  78.       } catch (Exception ex) { 
  79.        ex.printStackTrace(); 
  80.       } 
  81.      } 
  82.      System.out.println("Consuming value : " + queue.remove()); queue.notifyAll(); 
  83.     } 
  84.    } 
  85.   } 

如何在 Java 中正確使用 wait, notify 和 notifyAll – 以生產(chǎn)者消費者模型為例

為了更好地理解這個程序,我建議你在debug模式里跑這個程序。一旦你在debug模式下啟動程序,它會停止在PRODUCER或者 CONSUMER線程上,取決于哪個線程占據(jù)了CPU。因為兩個線程都有wait()的條件,它們一定會停止,然后你就可以跑這個程序然后看發(fā)生什么了 (很有可能它就會輸出我們以上展示的內(nèi)容)。你也可以使用Eclipse里的Step into和Step over按鈕來更好地理解多線程間發(fā)生的事情。

本文重點:

1. 你可以使用wait和notify函數(shù)來實現(xiàn)線程間通信。你可以用它們來實現(xiàn)多線程(>3)之間的通信。

2. 永遠在synchronized的函數(shù)或?qū)ο罄锸褂脀ait、notify和notifyAll,不然Java虛擬機會生成 IllegalMonitorStateException。

3. 永遠在while循環(huán)里而不是if語句下使用wait。這樣,循環(huán)會在線程睡眠前后都檢查wait的條件,并在條件實際上并未改變的情況下處理喚醒通知。

4. 永遠在多線程間共享的對象(在生產(chǎn)者消費者模型里即緩沖區(qū)隊列)上使用wait。

5. 基于前文提及的理由,更傾向用 notifyAll(),而不是 notify()。

如何在 Java 中正確使用 wait, notify 和 notifyAll – 以生產(chǎn)者消費者模型為例

這是關(guān)于Java里如何使用wait, notify和notifyAll的所有重點啦。你應(yīng)該只在你知道自己要做什么的情況下使用這些函數(shù),不然Java里還有很多其它的用來解決同步問題的方 案。例如,如果你想使用生產(chǎn)者消費者模型的話,你也可以使用BlockingQueue,它會幫你處理所有的線程安全問題和流程控制。如果你想要某一個線 程等待另一個線程做出反饋再繼續(xù)運行,你也可以使用CycliBarrier或者CountDownLatch。如果你只是想保護某一個資源的話,你也可 以使用Semaphore。

責(zé)任編輯:王雪燕 來源: ImportNew
相關(guān)推薦

2023-11-26 18:31:41

Linux信號

2021-10-25 09:00:37

Node.jsJS前端

2019-10-18 10:43:11

JPASpring Boot Flyway

2022-09-16 14:13:50

人工智能樓宇自動化

2022-06-13 07:33:57

socketReact組件

2022-02-21 08:20:24

Wait方法Notify

2025-01-09 08:49:36

Java并發(fā)編程

2021-11-05 15:10:28

UbuntuLinuxJAVA_HOME

2010-03-15 18:25:27

Java編程語言

2020-12-15 08:06:45

waitnotifyCondition

2010-03-15 19:11:39

Java多線程語句

2011-03-14 16:42:54

DB2 UDB監(jiān)控死鎖

2009-04-27 13:15:04

多線程方法run()

2010-06-18 13:14:26

SQL Server

2021-07-10 08:37:36

Notify機制Java

2022-08-10 13:12:04

Linuxcat命令

2010-08-20 11:16:39

DB2 優(yōu)化器

2023-12-26 11:56:14

Go通道編程

2010-07-15 09:47:09

SQL Server數(shù)

2023-10-12 07:35:45

面試線程通信
點贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 日韩欧美三区 | 日韩国产欧美在线观看 | 男女视频在线免费观看 | 国产精品爱久久久久久久 | xx性欧美肥妇精品久久久久久 | www.亚洲.com | 日韩中文字幕网 | 狠狠av| 国产精品不卡 | 午夜免费视频 | 四虎在线观看 | 国产91在线播放 | 午夜在线视频一区二区三区 | 日本色高清 | av男人的天堂在线 | 精品日本久久久久久久久久 | 91在线看 | 欧美日韩成人影院 | 成人午夜免费福利视频 | 国产精品自拍视频 | 九一视频在线播放 | 久久国产秒 | 一区二区三区欧美在线 | 国产综合精品一区二区三区 | 黄色大片观看 | 国产精品人人做人人爽 | 成人免费大片黄在线播放 | 久久精品亚洲国产 | 国产免费一区二区三区免费视频 | 欧美一区二区免费 | 免费在线观看黄视频 | 亚洲精品视频在线看 | 国产一级黄色网 | 国产精品久久久久久一级毛片 | 亚洲毛片一区二区 | 99久久精品免费看国产四区 | 在线免费观看黄视频 | 成人深夜福利在线观看 | 国产视频中文字幕 | 日一区二区 | 久久成人精品 |