亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

基于線程的wait和notify使用,生產(chǎn)消費(fèi)案例

 更新時(shí)間:2021年08月20日 09:48:07   作者:strive_day  
這篇文章主要介紹了基于線程的wait和notify使用,生產(chǎn)消費(fèi)案例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

多個(gè)線程可以相互競(jìng)爭(zhēng),也可以互相協(xié)作完成一件事情。

Object的相關(guān)方法

Object相關(guān)方法 描述
void wait() 讓當(dāng)前線程等待,如果沒有被喚醒,就一直等待
void wait(long timeout) 讓當(dāng)前線程等待指定毫秒值,如果到了指定的毫秒值自動(dòng)喚醒
void notify() 喚醒一個(gè)線程,喚醒的是當(dāng)前對(duì)象鎖下的一個(gè)線程
void notifyAll() 喚醒所有線程,喚醒的是當(dāng)前對(duì)象鎖下面的所有線程

這些方法一定要放在同步代碼塊中去使用,并且這些方法要通過鎖對(duì)象去調(diào)用【***】

案例:

生產(chǎn)方每生產(chǎn)一個(gè)產(chǎn)品就需要等待(通知)消費(fèi)方消費(fèi)完產(chǎn)品后才能繼續(xù)生產(chǎn)

消費(fèi)方每消費(fèi)一個(gè)產(chǎn)品就需要等待(通知)生產(chǎn)方去生產(chǎn)產(chǎn)品后才能繼續(xù)消費(fèi)。

案例圖解

生產(chǎn)方邏輯圖

在這里插入圖片描述

消費(fèi)方邏輯圖

在這里插入圖片描述

代碼實(shí)現(xiàn)

【注意】

notify、wait寫在同步代碼塊中,并且使用同一個(gè)對(duì)象(共有對(duì)象:倉庫)進(jìn)行操作。

this.cangku.wait() 和this.wait() 前一個(gè)使用的是倉庫對(duì)象 ,后一個(gè)使用的是當(dāng)前任務(wù)對(duì)象(使用后一個(gè)會(huì)造成死鎖)

//倉庫 - 唯一(鎖對(duì)象,任何對(duì)象都可以,用共有對(duì)象做鎖對(duì)象)
class CangKu { //當(dāng)作 鎖對(duì)象
    //定義一個(gè)變量體現(xiàn)數(shù)量
    public int productNum = 0;
}
//生產(chǎn)方和消費(fèi)方共用一個(gè)倉庫
//生產(chǎn)方
class ProductTask implements Runnable {
    private CangKu cangKu; //共用一個(gè)倉庫不能自己創(chuàng)建,由外部傳入
    public ProductTask(CangKu cangKu) { //通過構(gòu)造函數(shù)初始化
        this.cangKu = cangKu;
    }
    @Override
    public void run() {
        while (true) {
            //通知notify與等待wait必須寫在同步代碼塊中
            synchronized (this.cangKu) {//判斷是否有鎖可用,有就進(jìn)入
                if (this.cangKu.productNum == 0) {
                    ++this.cangKu.productNum;   //生產(chǎn)數(shù)目+1
                    System.out.println("生產(chǎn)了一個(gè)產(chǎn)品,當(dāng)前產(chǎn)品數(shù)目:" + this.cangKu.productNum);
                    //通知消費(fèi)者,必須用同一個(gè)鎖對(duì)象,不然會(huì)造成死鎖
                    this.cangKu.notify();
                } else {
                    //當(dāng)前還有存貨不用生產(chǎn),等待通知
                    try {
                        System.out.println("生產(chǎn)方等待中...");
                        this.cangKu.wait();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }//end if
            }//end synchronized 出房間釋放鎖
        }
    }
}
//消費(fèi)方
class ConsumerTask implements Runnable {
    private CangKu cangKu;
    public ConsumerTask(CangKu cangKu) {    //構(gòu)造方法
        this.cangKu = cangKu;
    }
    @Override
    public void run() {
        while (true) {
            synchronized (this.cangKu) {
                //判斷,倉庫是否為0
                if (this.cangKu.productNum == 0) {
                    try {
                        System.out.println("消費(fèi)方等待中...");
                        this.cangKu.wait();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                } else {
                    //有貨可以吃
                    -- this.cangKu.productNum ;
                    System.out.println("消費(fèi)了一個(gè)產(chǎn)品,當(dāng)前產(chǎn)品數(shù)目:" + this.cangKu.productNum);
                    //通知生產(chǎn)方生產(chǎn)產(chǎn)品
                    this.cangKu.notify();
                }//end if
            }//end synchronized
        }
    }
}
public class Wait_Notify_Demo {
    public static void main(String[] args) {
        //任務(wù)對(duì)象(生產(chǎn)方和消費(fèi)方共用一個(gè)倉庫)
        CangKu cangKu = new CangKu();
        ProductTask productTask = new ProductTask(cangKu);
        ConsumerTask consumerTask = new ConsumerTask(cangKu);
        //定義線程(用Executors線程池)
        ExecutorService pool = Executors.newFixedThreadPool(2);
        pool.submit(productTask);   //生產(chǎn)
        pool.submit(consumerTask);  //消費(fèi)
    }
}

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • Java中的序列化(Serializable)和反序列化

    Java中的序列化(Serializable)和反序列化

    這篇文章主要介紹了Java中的序列化(Serializable)和反序列化,?JAVA序列化與反序列化就是JAVA對(duì)象與一串字節(jié)流之間的相互轉(zhuǎn)換,?我們?cè)诔绦蛑袆?chuàng)建的JAVA對(duì)象只存在于JVM中,需要的朋友可以參考下
    2023-09-09
  • Java中的運(yùn)算符有哪些詳解

    Java中的運(yùn)算符有哪些詳解

    這篇文章主要給大家介紹了關(guān)于Java中運(yùn)算符有哪些的相關(guān)資料,包括算術(shù)運(yùn)算符、關(guān)系運(yùn)算符、邏輯運(yùn)算符、位運(yùn)算符、增量運(yùn)算符和自增/自減運(yùn)算符,文中通過代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2024-10-10
  • 詳解Java線程池如何統(tǒng)計(jì)線程空閑時(shí)間

    詳解Java線程池如何統(tǒng)計(jì)線程空閑時(shí)間

    這篇文章主要和大家分享一個(gè)面試題:Java線程池是怎么統(tǒng)計(jì)線程空閑時(shí)間?文中的示例代碼講解詳細(xì),對(duì)我們掌握J(rèn)ava有一定幫助,需要的可以參考一下
    2022-11-11
  • java 判斷l(xiāng)ist是否為空過程解析

    java 判斷l(xiāng)ist是否為空過程解析

    這篇文章主要介紹了java 判斷l(xiāng)ist是否為空過程解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-09-09
  • java動(dòng)態(tài)口令登錄實(shí)現(xiàn)過程詳解

    java動(dòng)態(tài)口令登錄實(shí)現(xiàn)過程詳解

    這篇文章主要介紹了java動(dòng)態(tài)口令登錄實(shí)現(xiàn)過程詳解,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-07-07
  • 詳析Spring中依賴注入的三種方式

    詳析Spring中依賴注入的三種方式

    在開發(fā)的過程中突然對(duì)Spring的依賴注入幾種方式出現(xiàn)混交,打算做個(gè)簡(jiǎn)單的小結(jié),方便大家和自己以后參考借鑒,如有總結(jié)不對(duì)的地方,請(qǐng)大家不吝指教!下面來一起看看吧。
    2016-09-09
  • Java高并發(fā)系統(tǒng)限流算法的實(shí)現(xiàn)

    Java高并發(fā)系統(tǒng)限流算法的實(shí)現(xiàn)

    這篇文章主要介紹了Java高并發(fā)系統(tǒng)限流算法的應(yīng)用,在開發(fā)高并發(fā)系統(tǒng)時(shí)有三把利器用來保護(hù)系統(tǒng):緩存、降級(jí)和限流,限流可以認(rèn)為服務(wù)降級(jí)的一種,限流是對(duì)系統(tǒng)的一種保護(hù)措施,需要的朋友可以參考下
    2022-05-05
  • Mybatis基礎(chǔ)概念與高級(jí)應(yīng)用小結(jié)

    Mybatis基礎(chǔ)概念與高級(jí)應(yīng)用小結(jié)

    這篇文章主要介紹了Mybatis基礎(chǔ)回顧與高級(jí)應(yīng)用,本文內(nèi)容有點(diǎn)小長(zhǎng),希望大家耐心閱讀,此文結(jié)合實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下
    2022-06-06
  • Springboot實(shí)現(xiàn)頁面間跳轉(zhuǎn)功能

    Springboot實(shí)現(xiàn)頁面間跳轉(zhuǎn)功能

    這篇文章主要介紹了Springboot實(shí)現(xiàn)頁面間跳轉(zhuǎn)功能,本文給大家分享兩種方式,方法一和方法二是不沖突的,但是通常情況下如果用方法二addViewControllers,需要把方法一所寫的Controller類給注釋掉,需要的朋友可以參考下
    2023-10-10
  • SpringBoot Security密碼加鹽實(shí)例

    SpringBoot Security密碼加鹽實(shí)例

    這篇文章主要為打擊介紹了SpringBoot Security密碼加鹽實(shí)例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-02-02

最新評(píng)論