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

Java并發(fā)編程:使用Wait和Notify方法的注意事項

開發(fā) 前端
save?方法負責向緩沖區(qū)添加數(shù)據(jù),然后執(zhí)行notify?方法來喚醒之前等待的線程。take方法負責檢查緩沖區(qū)是否為空。如果為空,線程進入等待狀態(tài);如果不為空,線程從緩沖區(qū)中取出數(shù)據(jù)。

在之前的講解線程狀態(tài)的文章中,我們提到了wait和notify方法可以讓線程在運行狀態(tài)和等待狀態(tài)之間轉換。在這篇文章中,我們將深入探討wait、notify和notifyAll方法在使用中的注意事項。我們主要從三個問題入手:

  • 為什么wait方法必須在synchronized保護的代碼中使用?
  • 為什么wait方法需要在循環(huán)操作中使用?
  • wait/notify和sleep方法有什么異同?

1. 為什么wait()方法必須在synchronized修飾的代碼中使用?

為了找到這個問題的答案,我們不妨反過來思考:如果不要求在synchronized代碼中使用wait方法,會出現(xiàn)什么問題呢?讓我們來看這段代碼。

public class QueueDemo {
    Queue<String> buffer = new LinkedList<String>();
    public void save(String data) {
        buffer.add(data);
        notify(); // 因為可能有線程在 take() 方法中等待
    }

    public String take() throws InterruptedException {
        while (buffer.isEmpty()) {
            wait();
        }
        return buffer.remove();
    }
}

在這段代碼中,有兩個方法。save方法負責向緩沖區(qū)添加數(shù)據(jù),然后執(zhí)行notify方法來喚醒之前等待的線程。take方法負責檢查緩沖區(qū)是否為空。如果為空,線程進入等待狀態(tài);如果不為空,線程從緩沖區(qū)中取出數(shù)據(jù)。

這段代碼沒有使用synchronized保護,可能會出現(xiàn)以下情況:

  • 首先,消費者線程調用take方法,并判斷buffer.isEmpty是否返回true。如果返回true,表示緩沖區(qū)為空,線程準備進入等待狀態(tài)。然而,在線程調用wait方法之前,它被可能已經(jīng)被掛起了,wait方法沒有執(zhí)行。
  • 此時,生產者線程開始運行,并執(zhí)行了整個save方法。它向緩沖區(qū)添加了數(shù)據(jù),并執(zhí)行了notify方法,但notify沒有效果,因為消費者線程的wait方法還沒有執(zhí)行,所以沒有線程在等待被喚醒。
  • 隨后,之前被掛起的消費者線程恢復執(zhí)行,并調用了wait方法,進入等待狀態(tài)。

出現(xiàn)這個問題的原因是這里的“判斷 - 執(zhí)行”不是原子操作,它在中間被中斷,是線程不安全的。

假設此時沒有更多的生產者進行生產,消費者可能會陷入無限等待,因為它錯過了save方法中的notify喚醒。

你可以模擬一個生產者線程和一個消費者線程分別調用這兩個方法:

public class QueueDemo2 {
    Queue<String> buffer = new LinkedList<>();
    public void save(String data) {
        System.out.println("Produce a data");
        buffer.add(data);
        notify(); // 因為可能有人在 take() 中等待
    }
    public String take() throws InterruptedException {
        System.out.println("Try to consume a data");
        while (buffer.isEmpty()) {
            wait();
        }
        return buffer.remove();
    }
    public static void main(String[] args) throws InterruptedException {
        QueueDemo2 queueDemo = new QueueDemo2();
        Thread producerThread = new Thread(() -> {
            queueDemo.save("Hello World!");
        });
        Thread consumerThread = new Thread(() -> {
            try {
                System.out.println(queueDemo.take());
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });
        consumerThread.start();
        producerThread.start();
    }
}

你可以嘗試執(zhí)行這段代碼,看看是否會出現(xiàn)之前提到的問題。

實際輸出如下:

Try to consume a data
Produce a data
Exception in thread "Thread-0" Exception in thread "Thread-1"
java.lang.IllegalMonitorStateException
    at java.lang.Object.notify(Native Method)
    at thread.basic.chapter4.QueueDemo2.save(QueueDemo2.java:13)
    at thread.basic.chapter4.QueueDemo2.lambda$main$0(QueueDemo2.java:28)
    at java.lang.Thread.run(Thread.java:748)
java.lang.IllegalMonitorStateException
    at java.lang.Object.wait(Native Method)
    at java.lang.Object.wait(Object.java:502)
    at thread.basic.chapter4.QueueDemo2.take(QueueDemo2.java:19)
    at thread.basic.chapter4.QueueDemo2.lambda$main$1(QueueDemo2.java:33)

根本沒有犯錯的機會。wait方法和notify方法在沒有synchronized保護的代碼塊中執(zhí)行時,會直接拋出java.lang.IllegalMonitorStateException異常。

修改代碼:

public class SyncQueueDemo2 {
    Queue<String> buffer = new LinkedList<>();
    public synchronized void save(String data) {
        System.out.println("Produce a data");
        buffer.add(data);
        notify(); // 因為可能有人在 take() 中等待
    }
    public synchronized String take() throws InterruptedException {
        System.out.println("Try to consume a data");
        while (buffer.isEmpty()) {
            wait();
        }
        return buffer.remove();
    }
    public static void main(String[] args) throws InterruptedException {
        SyncQueueDemo2 queueDemo = new SyncQueueDemo2();
        Thread producerThread = new Thread(() -> {
            queueDemo.save("Hello World!");
        });
        Thread consumerThread = new Thread(() -> {
            try {
                System.out.println(queueDemo.take());
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });
        consumerThread.start();
        producerThread.start();
    }
}

再次執(zhí)行代碼,輸出如下:

Produce a data
Try to consume a data
Hello World!

可以看到,生產的"Hello World!"已經(jīng)被成功消費并打印到控制臺。

2. 為什么wait方法需要在循環(huán)操作中使用?

線程調用wait方法后,可能會出現(xiàn)虛假喚醒(spurious wakeup)的情況,即線程在沒有被notify/notifyAll調用、沒有被中斷、也沒有超時的情況下被喚醒,這是我們不希望發(fā)生的情況。

雖然在真實環(huán)境中,虛假喚醒的概率非常小,但程序仍然需要在虛假喚醒的情況下保證正確性,因此需要使用while循環(huán)結構。

while (條件不滿足) {
    obj.wait();
}

這樣,即使線程被虛假喚醒,如果條件不滿足,wait會繼續(xù)執(zhí)行,從而消除虛假喚醒導致的風險。

3.wait/notify和sleep方法有什么異同?

wait方法和sleep方法的相同點如下:

  • 它們都可以阻塞線程。
  • 它們都可以響應中斷:如果在等待過程中收到中斷信號,它們會響應并拋出InterruptedException異常。

它們之間也有很多不同點:

  • wait方法必須在synchronized保護的代碼中使用,而sleep方法沒有這個要求。
  • 當sleep方法在synchronized代碼中執(zhí)行時,它不會釋放鎖,而wait方法會主動釋放鎖。
  • sleep方法需要定義一個時間,時間到期后線程會主動恢復。對于沒有參數(shù)的wait方法,它意味著永久等待,直到被中斷或喚醒,不會主動恢復。
  • wait和notify是Object類的方法,而sleep是Thread類的方法。

好了,這次的內容就到這里,下次再見!

責任編輯:武曉燕 來源: 程序猿技術充電站
相關推薦

2021-07-10 08:37:36

Notify機制Java

2022-09-23 09:25:04

代碼方法

2009-06-12 09:46:40

Java String

2010-03-15 18:25:27

Java編程語言

2009-09-01 17:25:33

初學C#編程

2010-11-26 16:27:01

MySQL使用變量

2009-08-27 10:40:56

Java路徑

2023-12-12 09:06:06

2011-06-23 11:15:25

SEO網(wǎng)站優(yōu)化

2010-08-12 09:39:26

FlexaddChil

2024-02-01 09:39:02

asyncawaitPromise

2011-05-26 11:22:04

SEO

2012-03-12 16:46:22

NoSQL數(shù)據(jù)庫

2011-07-28 17:29:22

HBaseShell

2015-08-05 09:33:21

Javawaitnotify

2010-01-21 11:30:10

2009-06-25 14:41:06

JavaBean

2009-06-11 17:52:08

JavaBean

2011-03-22 08:56:30

2021-12-20 23:22:46

Java開發(fā)升級
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 亚洲精品91 | 毛片一级片 | 久久国产精品一区二区三区 | 国产九九av| 91国内精精品久久久久久婷婷 | 日韩欧美二区 | 超碰免费在 | 成人精品国产免费网站 | 国产成人网 | a级大片免费观看 | 中文字幕一级毛片视频 | 国产激情亚洲 | 国产91精品久久久久久久网曝门 | 精品91| 久久国产一区二区 | 国产精品日产欧美久久久久 | 操皮视频 | 中文字幕高清在线 | 国产成人亚洲精品 | 欧美成人一区二区三区 | 免费观看色 | 人人性人人性碰国产 | 欧美www在线| 成人a在线观看 | 免费在线观看一区二区三区 | 日韩午夜网站 | 久久国产高清 | 精品一区二区久久久久久久网精 | 一区二区成人 | 色综合久久天天综合网 | 久久久久国产一区二区三区四区 | 一区二区三区国产精品 | 国产精品毛片无码 | 日韩色在线 | 在线色网址 | 国产香蕉视频在线播放 | 久草网站 | 99久久免费精品国产免费高清 | 情侣av | 色999视频| 91高清免费观看 |