Java通過(guò)notify和wait實(shí)現(xiàn)線程間的通信功能
一、前言
在軟件開(kāi)發(fā)中,線程是實(shí)現(xiàn)并發(fā)執(zhí)行的重要手段,然而,線程之間的協(xié)作與通信卻是開(kāi)發(fā)者必須重點(diǎn)考慮的挑戰(zhàn)之一。Java作為一種廣泛應(yīng)用于多線程編程的語(yǔ)言,提供了一套強(qiáng)大而靈活的機(jī)制,讓不同線程之間能夠優(yōu)雅地交替執(zhí)行、傳遞信息,以實(shí)現(xiàn)協(xié)調(diào)合作。
本文將深入探討Java中通過(guò)notify和wait實(shí)現(xiàn)線程間通信的機(jī)制。
二、notify 和 wait
2.1 wait
2.1.1 wait 基本介紹
通過(guò)源碼我們可以知道 wait 是 object 對(duì)象方法,用于實(shí)現(xiàn)線程間的等待和通知機(jī)制。當(dāng)一個(gè)線程調(diào)用wait()方法時(shí),它會(huì)釋放當(dāng)前所持有的對(duì)象鎖,并進(jìn)入等待狀態(tài),直到被其他線程調(diào)用相同對(duì)象上的notify()或notifyAll()方法喚醒。
public final void wait() throws InterruptedException {
wait(0);
}
2.1.2 wait 注意點(diǎn)
- 要想使用wait方法,當(dāng)前線程必須擁有對(duì)應(yīng) object 的 mointor,即必須要擁有對(duì)應(yīng)對(duì)象的鎖,否則會(huì)報(bào)
java.lang.IllegalMonitorStateException比如:
Object o = new Object(); o.wait(); //java.lang.IllegalMonitorStateException
- 當(dāng)調(diào)用
wait()方法的線程被另一個(gè)線程中斷時(shí),wait()方法會(huì)拋出InterruptedException異常。 - 線程也可能會(huì)在沒(méi)有收到通知、中斷或超時(shí)的情況下被喚醒,即所謂的虛假喚醒。雖然這種情況在實(shí)踐中很少發(fā)生,但應(yīng)用程序必須通過(guò)測(cè)試應(yīng)該導(dǎo)致線程被喚醒的條件來(lái)防范這種情況,如果條件不滿足,則繼續(xù)等待。換句話說(shuō),等待應(yīng)該總是在循環(huán)中發(fā)生,如下所示:
synchronized (obj) {
while (<condition does not hold>)
obj.wait(timeout);
... // Perform action appropriate to condition
}
2.1.3 wait 使用場(chǎng)景
wait()方法是在Java中用于線程間通信和同步的重要方法之一。它通常與synchronized關(guān)鍵字一起使用,用于在多線程中協(xié)調(diào)線程之間的操作。下面是wait()方法的一些常見(jiàn)使用場(chǎng)景:
- 協(xié)調(diào)多個(gè)線程的操作:
wait()方法通常與notify()和notifyAll()方法一起使用,用于在多線程之間協(xié)調(diào)操作。一個(gè)線程可以調(diào)用wait()進(jìn)入等待狀態(tài),等待其他線程調(diào)用相同對(duì)象的notify()或notifyAll()方法來(lái)喚醒它。 - 等待條件滿足:線程可以調(diào)用
wait()方法等待某個(gè)條件的滿足。例如,一個(gè)線程可能等待某個(gè)變量的值發(fā)生改變或者等待某個(gè)事件發(fā)生。 - 線程安全的隊(duì)列實(shí)現(xiàn):
wait()方法常用于實(shí)現(xiàn)線程安全的隊(duì)列。當(dāng)隊(duì)列為空時(shí),消費(fèi)者線程調(diào)用wait()等待生產(chǎn)者線程向隊(duì)列中添加元素;當(dāng)隊(duì)列已滿時(shí),生產(chǎn)者線程調(diào)用wait()等待消費(fèi)者線程從隊(duì)列中取走元素。 - 實(shí)現(xiàn)生產(chǎn)者-消費(fèi)者模型:
wait()方法在生產(chǎn)者-消費(fèi)者模型中起著重要作用。生產(chǎn)者向共享的緩沖區(qū)添加數(shù)據(jù)時(shí),如果緩沖區(qū)已滿,生產(chǎn)者線程調(diào)用wait()等待消費(fèi)者取走數(shù)據(jù);消費(fèi)者從緩沖區(qū)取數(shù)據(jù)時(shí),如果緩沖區(qū)為空,消費(fèi)者線程調(diào)用wait()等待生產(chǎn)者添加數(shù)據(jù)。 - 線程間通信:
wait()方法是線程間通信的重要手段之一。線程可以通過(guò)等待并喚醒的機(jī)制來(lái)實(shí)現(xiàn)信息的傳遞和協(xié)調(diào)。
2.1.4 wait 的執(zhí)行原理
- 當(dāng)線程調(diào)用
wait()方法時(shí),它會(huì)釋放當(dāng)前持有的對(duì)象鎖,使得其他線程可以訪問(wèn)這個(gè)對(duì)象并執(zhí)行同步代碼塊。 - 調(diào)用
wait()方法的線程會(huì)進(jìn)入對(duì)象的等待隊(duì)列,等待其他線程調(diào)用相同對(duì)象的notify()或notifyAll()方法來(lái)喚醒它。 - 當(dāng)另一個(gè)線程調(diào)用相同對(duì)象的
notify()方法或notifyAll()方法時(shí),等待隊(duì)列中的線程會(huì)被喚醒,然后競(jìng)爭(zhēng)對(duì)象的鎖。 - 喚醒的線程會(huì)嘗試重新獲取對(duì)象鎖,然后繼續(xù)執(zhí)行。
2.1.5 wait 使用
- wait()或者wait(0): 調(diào)用該方法的線程進(jìn)入WAITING狀態(tài),只有等待另外線程的通知或被中斷才會(huì)返回,需要注意調(diào)用 wait() 方法后,會(huì)釋放對(duì)象的鎖
- wait(long) 會(huì)釋放鎖,超時(shí)等待一段時(shí)間,這里的參數(shù)時(shí)間是毫秒,也就是等待長(zhǎng)達(dá)毫秒,如果沒(méi)有通知就超時(shí)返回
- wait(long,int) 會(huì)釋放鎖,該方法與 wait(long) 類似,多了一個(gè) nanos 參數(shù),這個(gè)參數(shù)表示額外時(shí)間(以納秒為單位,范圍是 0-999999), 所以超時(shí)的時(shí)間還需要加上 nanos 納秒。
2.2 notify
首先我們需要知道 nofity 和 notifyall 基本上是等價(jià)的,如沒(méi)有特別標(biāo)明,nofity 和 nofityall 是一樣的
2.2.1 notify 基本介紹
通過(guò)源碼我們可以知道 notify 是 object 對(duì)象方法。
notify()是線程間通信的一種機(jī)制,用于喚醒在當(dāng)前對(duì)象上等待的一個(gè)線程。當(dāng)一個(gè)線程調(diào)用notify()方法時(shí),它會(huì)喚醒正在該對(duì)象上等待的單個(gè)線程(如果有多個(gè)線程在等待,系統(tǒng)無(wú)法確定哪個(gè)線程會(huì)被喚醒,因?yàn)檫x擇是隨機(jī)的)。這個(gè)被喚醒的線程將從等待狀態(tài)變?yōu)榭蛇\(yùn)行狀態(tài),但并不意味著立即獲得對(duì)象的鎖。
public final native void notify();
當(dāng)一個(gè)線程調(diào)用notifyAll()時(shí),它會(huì)喚醒在當(dāng)前對(duì)象上等待的所有線程,使它們從等待狀態(tài)轉(zhuǎn)變?yōu)榭蛇\(yùn)行狀態(tài)。這樣,所有等待中的線程都有機(jī)會(huì)爭(zhēng)取獲取對(duì)象鎖,在某個(gè)線程獲得鎖后,它們會(huì)競(jìng)爭(zhēng)執(zhí)行。
2.2.2 notify 注意點(diǎn)
下面是關(guān)于notify()方法的一些重要點(diǎn):
- 使用條件變量:
notify()方法通常與條件變量一起使用,用于線程間的協(xié)作。一個(gè)線程等待某個(gè)條件變?yōu)檎?,另一個(gè)線程在某種情況下會(huì)改變條件,并調(diào)用notify()來(lái)通知等待的線程。 - notifyAll()方法:與
notify()不同的是,notifyAll()方法喚醒在當(dāng)前對(duì)象上等待的所有線程,而不僅僅是一個(gè)線程。這樣做可以避免遺漏任何等待中的線程,但同一時(shí)間獲取鎖的只會(huì)有一個(gè)線程。 - Object類中的方法:
notify()方法是Object類的一個(gè)方法,因此任何Java對(duì)象都可以調(diào)用notify()和wait()方法來(lái)進(jìn)行線程間的通信。 - 必須在同步塊中調(diào)用:為了調(diào)用
notify()方法,必須在同步塊(synchronized塊)中對(duì)包含該對(duì)象的鎖進(jìn)行操作。 - 隨機(jī)性:當(dāng)多個(gè)線程在同一個(gè)對(duì)象上等待時(shí),調(diào)用
notify()會(huì)隨機(jī)選擇一個(gè)線程喚醒,因此不能確定哪個(gè)線程會(huì)被喚醒。 - 喚醒等待線程:被喚醒的線程會(huì)嘗試重新獲取對(duì)象鎖,一旦獲得鎖,它會(huì)從
wait()方法返回,并繼續(xù)執(zhí)行后續(xù)代碼。
2.2.3 nofityall
以下是關(guān)于notifyAll()方法的詳細(xì)介紹:
- 喚醒所有等待線程:
notifyAll()方法被調(diào)用時(shí),會(huì)喚醒在當(dāng)前對(duì)象上等待的所有線程,使它們從阻塞狀態(tài)轉(zhuǎn)變?yōu)榫途w狀態(tài)。這樣,所有等待中的線程都有機(jī)會(huì)去競(jìng)爭(zhēng)獲取對(duì)象的鎖。 - 解決等待問(wèn)題:
notifyAll()通常用于解決多線程之間的通信和協(xié)調(diào)問(wèn)題。當(dāng)某個(gè)條件得到滿足時(shí),調(diào)用notifyAll()可以通知所有等待該條件的線程。 - 謹(jǐn)慎使用:需要謹(jǐn)慎使用
notifyAll(),因?yàn)閱拘阉芯€程可能會(huì)導(dǎo)致競(jìng)爭(zhēng)和性能問(wèn)題。在某些情況下,如果只有一個(gè)線程是合適的接收方,那么使用notify()來(lái)喚醒特定線程會(huì)更有效率。 - 必須在同步塊中調(diào)用:就像
notify()方法一樣,notifyAll()也必須在同步塊(synchronized塊)內(nèi)調(diào)用,以確保在對(duì)象上同步。通過(guò)同步塊,確保在調(diào)用notifyAll()時(shí),持有對(duì)象的鎖。 - 競(jìng)爭(zhēng)獲取鎖:被喚醒的線程會(huì)嘗試獲取對(duì)象的鎖,一旦獲得鎖,它們將從
wait()方法返回,并開(kāi)始競(jìng)爭(zhēng)執(zhí)行。 - 釋放鎖:調(diào)用
notifyAll()并不會(huì)釋放對(duì)象鎖,它僅喚醒等待的線程,這些線程會(huì)在合適的時(shí)機(jī)爭(zhēng)奪鎖。
2.2.4 notify 使用場(chǎng)景
notify()的使用場(chǎng)景:
- 單個(gè)喚醒:當(dāng)只需喚醒等待隊(duì)列中的一個(gè)線程時(shí),適合使用
notify()。這種情況下,最多只有一個(gè)等待線程被喚醒,選擇被喚醒的線程是不確定的。 - 資源更新:當(dāng)某個(gè)共享資源的狀態(tài)發(fā)生變化時(shí),并且只需要通知一個(gè)線程來(lái)處理這種變化時(shí),可以使用
notify()。
notifyAll()的使用場(chǎng)景:
- 多個(gè)喚醒:當(dāng)需要喚醒所有等待線程來(lái)處理某個(gè)共享資源的變化時(shí),應(yīng)該使用
notifyAll()。這種情況下,所有等待線程都會(huì)被喚醒。 - 條件變更:當(dāng)共享資源的狀態(tài)發(fā)生變化對(duì)多個(gè)線程都有影響時(shí),可以使用
notifyAll()來(lái)通知所有等待線程,因?yàn)槎鄠€(gè)線程可能對(duì)資源的狀態(tài)變化做出不同的響應(yīng)。 - 避免競(jìng)爭(zhēng)問(wèn)題:在某些情況下,使用
notifyAll()可以避免因?yàn)橹粏拘岩粋€(gè)線程而導(dǎo)致的競(jìng)爭(zhēng)問(wèn)題,確保所有等待線程都能及時(shí)做出響應(yīng)。
2.2.5 notify 的執(zhí)行原理
notify()方法的執(zhí)行原理:
- 當(dāng)一個(gè)線程調(diào)用對(duì)象的
notify()方法時(shí),這個(gè)對(duì)象的監(jiān)視器被激活。 - 在等待該對(duì)象的線程中,會(huì)有一個(gè)線程從等待隊(duì)列中被選中,進(jìn)入到鎖定狀態(tài),并嘗試重新獲取對(duì)象的鎖。
- 一旦獲取到鎖,該線程會(huì)從
wait()方法返回,并繼續(xù)執(zhí)行。 - 其他等待該對(duì)象的線程仍然處于等待狀態(tài),需要重新競(jìng)爭(zhēng)獲取對(duì)象的鎖來(lái)繼續(xù)執(zhí)行。
notifyAll()方法的執(zhí)行原理:
- 當(dāng)一個(gè)線程調(diào)用對(duì)象的
notifyAll()方法時(shí),所有等待該對(duì)象的線程將被喚醒。 - 被喚醒的線程會(huì)一起競(jìng)爭(zhēng)對(duì)象的鎖,以便能夠繼續(xù)執(zhí)行。
- 每個(gè)被喚醒的線程嘗試重新獲取鎖,一旦成功獲取到鎖,它們會(huì)從
wait()方法返回,并接著執(zhí)行后續(xù)代碼。
2.2.6 notify和notifyAll 注意事項(xiàng)和要點(diǎn):
- 使用 notify和notifyAll 時(shí)需要先對(duì)調(diào)用的對(duì)象加鎖,否則會(huì)報(bào)錯(cuò)
notify()方法和notifyAll()方法必須在同步塊中(synchronized塊)調(diào)用,以確保在調(diào)用這些方法時(shí)對(duì)象的鎖處于正確狀態(tài)。- 被喚醒的線程在獲取到鎖并從
wait()方法返回后,可能需要檢查等待期間的條件是否發(fā)生了變化,從而決定是否繼續(xù)執(zhí)行。
三、wait/nofity 經(jīng)典使用方式
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.concurrent.TimeUnit;
public class WaitNotify {
static boolean flag = true;
static Object lock = new Object();
public static void main(String[] args) {
Thread waitThread = new Thread(new Wait(), "waitThread");
waitThread.start();
try {
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
e.printStackTrace();
}
new Thread(new Notify(), "notifyThread").start();
}
//wait
static class Wait implements Runnable {
public void run() {
synchronized (lock) {
while (flag) {
try {
System.out.println(Thread.currentThread() + " flag is true" + new SimpleDateFormat("HH:mm:ss").format(new Date()));
//沒(méi)有釋放資源 wait 等待
lock.wait();
} catch (InterruptedException e) {
}
}
System.out.println(Thread.currentThread() + " flag is false" + new SimpleDateFormat("HH:mm:ss").format(new Date()));
}
}
}
static class Notify implements Runnable {
public void run() {
synchronized (lock) {
while (flag) {
System.out.println(Thread.currentThread() + "hold lock notify" + new SimpleDateFormat("HH:mm:ss").format(new Date()));
//資源準(zhǔn)備
好了,nofity
lock.notifyAll();
flag = false;
try {
TimeUnit.SECONDS.sleep(5);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
synchronized (lock) {
System.out.println(Thread.currentThread() + "hold lock again. notify" + new SimpleDateFormat("HH:mm:ss").format(new Date()));
try {
TimeUnit.SECONDS.sleep(5);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
}
這是最經(jīng)典的等待/通知的范式:但資源沒(méi)有準(zhǔn)備好時(shí),wait 等待,當(dāng)資源準(zhǔn)備好了,notify 通知。
四、總結(jié)
文章詳細(xì)講解了Java中wait和notify方法的使用方法和注意事項(xiàng),通過(guò)這些方法實(shí)現(xiàn)線程間的協(xié)調(diào)與通信,并列舉了常見(jiàn)的使用場(chǎng)景,如協(xié)調(diào)多個(gè)線程操作、實(shí)現(xiàn)生產(chǎn)者-消費(fèi)者模型等。此外,文章還強(qiáng)調(diào)了wait和notify方法必須在同步塊中調(diào)用,以確保線程安全。
以上就是Java通過(guò)notify和wait實(shí)現(xiàn)線程間的通信功能的詳細(xì)內(nèi)容,更多關(guān)于Java notify wait線程通信的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
IDEA 當(dāng)前在線人數(shù)和歷史訪問(wèn)量的示例代碼
這篇文章主要介紹了IDEA 當(dāng)前在線人數(shù)和歷史訪問(wèn)量的實(shí)例代碼,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-08-08
java實(shí)現(xiàn)基于Tcp的socket聊天程序
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)基于Tcp的socket聊天程序,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-07-07
IntelliJ IDEA啟動(dòng)錯(cuò)誤:插件沖突處理的解決方案
在使用 IntelliJ IDEA 進(jìn)行開(kāi)發(fā)時(shí),我們可能會(huì)遇到各種啟動(dòng)錯(cuò)誤,本文將詳細(xì)介紹一種常見(jiàn)的錯(cuò)誤:插件沖突,并提供解決方案,文中通過(guò)圖文和代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,需要的朋友可以參考下2025-02-02
SpringCloud3.x集成BigQuery的代碼實(shí)現(xiàn)
Google BigQuery 是一種高性能、可應(yīng)用于大數(shù)據(jù)分析的公主云數(shù)據(jù)庫(kù)服務(wù),Spring Cloud 提供了完善的工具和核心功能,可以進(jìn)行泛動(dòng)分布應(yīng)用構(gòu)建,本文給大家介紹了SpringCloud3.x集成BigQuery的代碼實(shí)現(xiàn),需要的朋友可以參考下2025-01-01
一文詳解Java中的反射與new創(chuàng)建對(duì)象
Java中的反射(Reflection)和使用new關(guān)鍵字創(chuàng)建對(duì)象是兩種不同的對(duì)象創(chuàng)建方式,各有優(yōu)缺點(diǎn)和適用場(chǎng)景,本文小編給大家詳細(xì)介紹了Java中的反射與new創(chuàng)建對(duì)象,感興趣的小伙伴跟著小編一起來(lái)看看吧2024-07-07
在 Spring Boot 中集成 MinIO 對(duì)象存儲(chǔ)
MinIO 是一個(gè)開(kāi)源的對(duì)象存儲(chǔ)服務(wù)器,專注于高性能、分布式和兼容S3 API的存儲(chǔ)解決方案,本文將介紹如何在 Spring Boot 應(yīng)用程序中集成 MinIO,以便您可以輕松地將對(duì)象存儲(chǔ)集成到您的應(yīng)用中,需要的朋友可以參考下2023-09-09

