聊聊Object類(lèi)中的wait()和notify()方法
Object類(lèi)中的wait()和notify()方法
一、特點(diǎn)
wait()和notify()方法并不是線程對(duì)象的方法,是Java中任何一個(gè)Java對(duì)象都有的方法,并不特殊。
二、wait()方法的作用
Object obj = new Object(); obj.wait();
表示:obj.wait();方法的調(diào)用,會(huì)讓“當(dāng)前線程(正在obj對(duì)象上活動(dòng)的線程)”進(jìn)入等待狀態(tài)。
三、notify()方法的作用
Object obj = new Object(); obj.notify();
表示:?jiǎn)拘颜趏bj對(duì)象上等待的線程。
補(bǔ)充:
Object obj = new Object(); obj.notifyAll();
表示:?jiǎn)拘颜趏bj對(duì)象上等待的所有線程。
四、wait()和notify()的使用
wait()和notify()方法都是建立在synchronized線程同步的基礎(chǔ)之上
重點(diǎn)
obj.wait()方法會(huì)讓正在obj對(duì)象上活動(dòng)的當(dāng)前線程進(jìn)入等待狀態(tài),并且釋放之前占有的obj對(duì)象的鎖。
obj.notify()方法只會(huì)通知,不會(huì)釋放之前占有的obj對(duì)象的鎖。
object中的wait和notify細(xì)節(jié)
wait
jdk源碼:
重點(diǎn)看下劃線的地方,是不是有些不理解。有個(gè)印象我們繼續(xù)往下看。
public class resourse { private Integer number = 0 ; /** * 用if為啥不行 * 1:首先一點(diǎn)我們要搞清楚 wait操作會(huì)釋放鎖 * 2:想想這種情況,當(dāng)一個(gè)生產(chǎn)者線程執(zhí)行的時(shí)候 if number!=0 (此時(shí)的number為1 ) 就會(huì)發(fā)生阻塞 這時(shí)候 * 釋放出鎖 這時(shí)候又一個(gè)生產(chǎn)者進(jìn)程進(jìn)來(lái)又會(huì)被wait住.然后一個(gè)生產(chǎn)者進(jìn)程進(jìn)來(lái),消費(fèi)了一個(gè) 但是notifyall 將所有的進(jìn)程 * 都解開(kāi)了 。。那兩個(gè)生產(chǎn)者進(jìn)程就會(huì)直接運(yùn)行if后面的東西并沒(méi)有被拉回來(lái)重新判斷一下。這樣就造成了number的值變成2.同理number * 變成負(fù)數(shù)也是有可能的(兩個(gè)消費(fèi)者進(jìn)程先進(jìn)來(lái)都堵塞). * * * * * @throws Exception */ public synchronized void produce () throws Exception { // 判斷 這里用while 用if多與兩個(gè)線程容易出錯(cuò) // 不等于0就要等待消費(fèi)者消費(fèi)完 if(number!=0) { this.wait(); } // 干活 number++; System.out.println(Thread.currentThread().getName() + "的資源數(shù)為:" + number.toString()); // 釋放 this.notifyAll(); } //wait操作會(huì)釋放鎖 public synchronized void consumer () throws Exception { // 等于零就要等待生產(chǎn)者生產(chǎn) if (number == 0) { this.wait(); } // 消費(fèi) number--; System.out.println(Thread.currentThread().getName() + "的資源數(shù)為:" + number.toString()); // 釋放 this.notifyAll(); } }
調(diào)用者。這里開(kāi)啟了兩個(gè)生產(chǎn)者和兩個(gè)消費(fèi)者線程。生產(chǎn)者線程都執(zhí)行100次的produce,消費(fèi)者線程都執(zhí)行100次的consumer
public class main { public static void main (String[] args) { resourse resourse = new resourse(); new Thread(()->{ try{ for (int i=0 ; i<100;i++) resourse.produce(); }catch (Exception e) { e.printStackTrace(); } }).start(); new Thread(()->{ try{ for (int i=0 ; i<100;i++) resourse.produce(); }catch (Exception e) { e.printStackTrace(); } }).start(); new Thread(()->{ try{ for (int i=0 ; i<100;i++) resourse.consumer(); }catch (Exception e) { e.printStackTrace(); } }).start(); new Thread(()->{ try{ for (int i=0 ; i<100;i++) resourse.consumer(); }catch (Exception e) { e.printStackTrace(); } }).start(); } }
我要說(shuō)的重點(diǎn)是:
這樣設(shè)計(jì)真的合理嗎?
jdk中為啥要規(guī)定我們要使用while,而不是if?
用if為啥不行
1:首先一點(diǎn)我們要搞清楚 wait操作會(huì)釋放鎖
2:想想這種情況,當(dāng)一個(gè)生產(chǎn)者線程執(zhí)行的時(shí)候 if number!=0 (此時(shí)的number為1 ) 就會(huì)發(fā)生阻塞 這時(shí)候釋放出鎖 這時(shí)候又一個(gè)生產(chǎn)者進(jìn)程進(jìn)來(lái)又會(huì)被wait住.然后一個(gè)生產(chǎn)者進(jìn)程進(jìn)來(lái),消費(fèi)了一個(gè) 但是notifyall 將所有的進(jìn)程都解開(kāi)了 。。那兩個(gè)生產(chǎn)者進(jìn)程就會(huì)直接運(yùn)行if后面的東西并沒(méi)有被拉回來(lái)重新判斷一下。這樣就造成了number的值變成2.同理number變成負(fù)數(shù)也是有可能的(兩個(gè)消費(fèi)者進(jìn)程先進(jìn)來(lái)都堵塞).
這個(gè)問(wèn)題絕對(duì)是干貨,在工作中絕對(duì)會(huì)遇到生產(chǎn)者消費(fèi)者問(wèn)題,不少程序員會(huì)在這個(gè)地方踩坑。面試的時(shí)候如果考到你這個(gè)地方,你能解答出深層原理來(lái)。相信面試官會(huì)高看你一眼。
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
- Java多線程通信wait()和notify()代碼實(shí)例
- Object類(lèi)wait及notify方法原理實(shí)例解析
- Java object wait notify notifyAll代碼解析
- Java多線程中的wait/notify通信模式實(shí)例詳解
- 分析java并發(fā)中的wait notify notifyAll
- java wait()/notify() 實(shí)現(xiàn)生產(chǎn)者消費(fèi)者模式詳解
- 基于線程的wait和notify使用,生產(chǎn)消費(fèi)案例
- Java如何正確的使用wait-notify方法你知道嗎
- 為什么wait和notify必須放在synchronized中使用
相關(guān)文章
圖解Java?ReentrantLock的條件變量Condition機(jī)制
想必大家都使用過(guò)wait()和notify()這兩個(gè)方法把,他們主要用于多線程間的協(xié)同處理。而RenentrantLock也支持這樣條件變量的能力,而且相對(duì)于synchronized?更加強(qiáng)大,能夠支持多個(gè)條件變量,本文就來(lái)詳細(xì)說(shuō)說(shuō)2022-10-10IDEA+Maven搭建Spring環(huán)境的詳細(xì)教程
這篇文章主要介紹了IDEA+Maven搭建Spring環(huán)境的詳細(xì)教程,本文通過(guò)圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-11-11Java獲取Excel中圖片所在的行和列坐標(biāo)位置
這篇文章主要介紹了Java獲取Excel中圖片所在的行和列坐標(biāo)位置,文章圍繞主題展開(kāi)詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的朋友可以參考一下2022-08-08java反射拼接方法名動(dòng)態(tài)執(zhí)行方法實(shí)例
下面小編就為大家?guī)?lái)一篇java反射拼接方法名動(dòng)態(tài)執(zhí)行方法實(shí)例。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-08-08解決微服務(wù)feign調(diào)用添加token的問(wèn)題
這篇文章主要介紹了解決微服務(wù)feign調(diào)用添加token的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-06-06學(xué)習(xí)SpringMVC——如何獲取請(qǐng)求參數(shù)詳解
本篇文章主要介紹了SpringMVC——如何獲取請(qǐng)求參數(shù)詳解,詳細(xì)的介紹了每種參數(shù)注解的用法。具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下。2016-12-12