基于线程的wait和notify使用,生产消费案例

目录
  • 案例图解
    • 生产方逻辑图
    • 消费方逻辑图
  • 代码实现

多个线程可以相互竞争,也可以互相协作完成一件事情。

Object的相关方法

Object相关方法 描述
void wait() 让当前线程等待,如果没有被唤醒,就一直等待
void wait(long timeout) 让当前线程等待指定毫秒值,如果到了指定的毫秒值自动唤醒
void notify() 唤醒一个线程,唤醒的是当前对象锁下的一个线程
void notifyAll() 唤醒所有线程,唤醒的是当前对象锁下面的所有线程

这些方法一定要放在同步代码块中去使用,并且这些方法要通过锁对象去调用【***】

案例:

生产方每生产一个产品就需要等待(通知)消费方消费完产品后才能继续生产

消费方每消费一个产品就需要等待(通知)生产方去生产产品后才能继续消费。

案例图解

生产方逻辑图

消费方逻辑图

代码实现

【注意】

notify、wait写在同步代码块中,并且使用同一个对象(共有对象:仓库)进行操作。

this.cangku.wait() 和this.wait() 前一个使用的是仓库对象 ,后一个使用的是当前任务对象(使用后一个会造成死锁)

//仓库 - 唯一(锁对象,任何对象都可以,用共有对象做锁对象)
class CangKu { //当作 锁对象
    //定义一个变量体现数量
    public int productNum = 0;
}
//生产方和消费方共用一个仓库
//生产方
class ProductTask implements Runnable {
    private CangKu cangKu; //共用一个仓库不能自己创建,由外部传入
    public ProductTask(CangKu cangKu) { //通过构造函数初始化
        this.cangKu = cangKu;
    }
    @Override
    public void run() {
        while (true) {
            //通知notify与等待wait必须写在同步代码块中
            synchronized (this.cangKu) {//判断是否有锁可用,有就进入
                if (this.cangKu.productNum == 0) {
                    ++this.cangKu.productNum;   //生产数目+1
                    System.out.println("生产了一个产品,当前产品数目:" + this.cangKu.productNum);
                    //通知消费者,必须用同一个锁对象,不然会造成死锁
                    this.cangKu.notify();
                } else {
                    //当前还有存货不用生产,等待通知
                    try {
                        System.out.println("生产方等待中...");
                        this.cangKu.wait();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }//end if
            }//end synchronized 出房间释放锁
        }
    }
}
//消费方
class ConsumerTask implements Runnable {
    private CangKu cangKu;
    public ConsumerTask(CangKu cangKu) {    //构造方法
        this.cangKu = cangKu;
    }
    @Override
    public void run() {
        while (true) {
            synchronized (this.cangKu) {
                //判断,仓库是否为0
                if (this.cangKu.productNum == 0) {
                    try {
                        System.out.println("消费方等待中...");
                        this.cangKu.wait();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                } else {
                    //有货可以吃
                    -- this.cangKu.productNum ;
                    System.out.println("消费了一个产品,当前产品数目:" + this.cangKu.productNum);
                    //通知生产方生产产品
                    this.cangKu.notify();
                }//end if
            }//end synchronized
        }
    }
}
public class Wait_Notify_Demo {
    public static void main(String[] args) {
        //任务对象(生产方和消费方共用一个仓库)
        CangKu cangKu = new CangKu();
        ProductTask productTask = new ProductTask(cangKu);
        ConsumerTask consumerTask = new ConsumerTask(cangKu);
        //定义线程(用Executors线程池)
        ExecutorService pool = Executors.newFixedThreadPool(2);
        pool.submit(productTask);   //生产
        pool.submit(consumerTask);  //消费
    }
}

以上为个人经验,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

  • Object类wait及notify方法原理实例解析

    Object类中的wait和notify方法(生产者和消费者模式) 不是通过线程调用 wait(): 让正在当前对象上活动的线程进入等待状态,无期限等待,直到被唤醒为止 notify(): 让正在当前对象上等待的线程唤醒 notifyAll(): 唤醒当前对象上处于等待的所有线程 生产者和消费者模式 生产线程和消费线程达到均衡 wait方法和notify方法建立在synchronized线程同步的基础之上 wait方法: 释放当前对象占有的锁 notify方法: 通知,不会释放锁 实现生产者和消

  • Java多线程通信wait()和notify()代码实例

    1.wait()方法和sleep()方法: wait()方法在等待中释放锁:sleep()在等待的时候不会释放锁,抱着锁睡眠. 2.notify(): 随机唤醒一个线程,将等待队列中的一个等待线程从等待队列中移到同步队列中. 代码如下 public class Demo_Print { public static void main(String[] args) { Print p = new Print(); new Thread() { public void run() { while (

  • Java多线程中wait、notify、notifyAll使用详解

    基础知识 首先我们需要知道,这几个都是Object对象的方法.换言之,Java中所有的对象都有这些方法. public final native void notify(); public final native void notifyAll(); public final native void wait(long timeout) throws InterruptedException; public final void wait() throws InterruptedExceptio

  • Java多线程中的wait/notify通信模式实例详解

    前言 最近在看一些JUC下的源码,更加意识到想要学好Java多线程,基础是关键,比如想要学好ReentranLock源码,就得掌握好AQS源码,而AQS源码中又有很多Java多线程经典的一些应用:再比如看了线程池的核心源码实现,又学到了很多核心实现,其实这些都可以提出来慢慢消化并变成自己的知识点,今天这个Java等待/通知模式其实是Thread.join()实现的关键,还有线程池工作线程中线程跟线程之间的通信的核心所在,故在此为了加深理解,做此记录! 参考资料<Java并发编程艺术>(电子PD

  • 基于线程的wait和notify使用,生产消费案例

    目录 案例图解 生产方逻辑图 消费方逻辑图 代码实现 多个线程可以相互竞争,也可以互相协作完成一件事情. Object的相关方法 Object相关方法 描述 void wait() 让当前线程等待,如果没有被唤醒,就一直等待 void wait(long timeout) 让当前线程等待指定毫秒值,如果到了指定的毫秒值自动唤醒 void notify() 唤醒一个线程,唤醒的是当前对象锁下的一个线程 void notifyAll() 唤醒所有线程,唤醒的是当前对象锁下面的所有线程 这些方法一定要

  • 基于线程、并发的基本概念(详解)

    什么是线程? 提到"线程"总免不了要和"进程"做比较,而我认为在Java并发编程中混淆的不是"线程"和"进程"的区别,而是"任务(Task)".进程是表示资源分配的基本单位.而线程则是进程中执行运算的最小单位,即执行处理机调度的基本单位.关于"线程"和"进程"的区别耳熟能详,说来说去就一句话:通常来讲一个程序有一个进程,而一个进程可以有多个线程. 但是"任务

  • java基于线程池和反射机制实现定时任务完整实例

    本文实例讲述了java基于线程池和反射机制实现定时任务的方法.分享给大家供大家参考,具体如下: 主要包括如下实现类: 1. Main类: 任务执行的入口: 调用main方法,开始加载任务配置并执行任务 package com.yanek.task; import java.util.List; import java.util.concurrent.Executors; import java.util.concurrent.ScheduledExecutorService; import ja

  • Java基于线程实现带有滚动效果的Label标签实例

    本文实例讲述了Java基于线程实现带有滚动效果的Label标签.分享给大家供大家参考.具体如下: import java.awt.Graphics; import javax.swing.JFrame; import javax.swing.JLabel; import javax.swing.JPanel; /** * Java中用线程实现带有滚动效果的Label标签 */ public class Test extends JFrame { private static final long

  • 浅谈线程通信wait,notify作用

    线程通信的目的是为了能够让线程之间相互发送信号.另外,线程通信还能够使得线程等待其它线程的信号,比如,线程B可以等待线程A的信号,这个信号可以是线程A已经处理完成的信号 Wait()方法 -中断方法的执行,使本线程等待,暂时让出cpu的使用权,并允许其他线程使用这个同步方法 Notify()方法 -唤醒由于使用这个同步方而处于等待线程的某一个结束等待 Notifyall()方法 唤醒所有由于使用这个同步方法而处于等待的线程结束等待 什么时候使用wait方法 当一个线程使用的同步方法中用到某个变量

  • 基于线程池的工作原理与源码解读

    随着cpu核数越来越多,不可避免的利用多线程技术以充分利用其计算能力.所以,多线程技术是服务端开发人员必须掌握的技术. 线程的创建和销毁,都涉及到系统调用,比较消耗系统资源,所以就引入了线程池技术,避免频繁的线程创建和销毁. 在Java用有一个Executors工具类,可以为我们创建一个线程池,其本质就是new了一个ThreadPoolExecutor对象.线程池几乎也是面试必考问题.本节结合源代码,说说ThreadExecutor的工作原理 一.线程池创建 先看一下ThreadPoolExec

  • Go语言kafka生产消费消息实例搬砖

    目录 kafka go库 注意 例子 kafka go库 kafka go客户端官方目前没有提供,但在github有2个非常流行的库 星星较多,网上案例也多 https://github.com/Shopify/sarama confluent官网提供的库 https://github.com/confluentinc/confluent-kafka-go 这里使用sarama,因为星星多,案例多,方便快速上手 注意 如果kafka版本在2.2以下,需要在go.mod里面将sarama版本改为g

  • 如何基于线程池提升request模块效率

    普通方法:爬取梨视频 import re import time import random import requests from lxml import etree start_time = time.time() url = "https://www.pearvideo.com/category_3" headers = { "User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) Apple

  • 基于Apache组件分析对象池原理的实现案例分析

    目录 一.设计与原理 1.基础案例 2.接口设计 1.1 PooledObjectFactory 接口 1.2 ObjectPool 接口 1.3 PooledObject 接口 3.运行原理 二.构造分析 1.对象池 2.双端队列 三.对象管理 1.添加对象 2.借用对象 3.归还对象 4.对象状态 四.Redis应用 1.配置管理 2.源码分析 2.1 配置转换 2.2 对象池构造 2.3 对象管理 五.参考源码 池塘里养:Object: 一.设计与原理 1.基础案例 首先看一个基于comm

  • JS基于开关思想实现的数组去重功能【案例】

    本文实例讲述了JS基于开关思想实现的数组去重功能.分享给大家供大家参考,具体如下: 场景: 比如给你一个数组var Arr = [ 25, 70, 60, 70, 65, 65, 80 ],最终要得到去重后的新数组[25,70,60,65,80]. 思路: 1.先定义个空的新数组newArr; 2.遍历旧数组Arr 中所有元素; 3.定义一个isZai变量表示开关,默认为false状态; 4.遍历新数组所有元素与旧数组元素比较(Arr[ i ] == newArr[ j ]); 5.如果该元素存

随机推荐