当前位置: 首页 > news >正文

Java基础-多线程JUC-生产者和消费者

1. 生产者与消费者

在这里插入图片描述

实现线程轮流交替执行的结果;
实现线程休眠和唤醒均要使用到锁对象;
修改标注位(foodFlag);
代码实现:

public class demo11 {public static void main(String[] args) {/*** 需求:完成消费者和生产者(等待唤醒机制)的代码*      实现线程轮流交替执行的结果*/Cooker cooker = new Cooker();Foodie foodie = new Foodie();cooker.setName("厨师");foodie.setName("吃货");cooker.start();foodie.start();}
}
class Cooker extends Thread {@Overridepublic void run() {/*** 1.循环* 2.同步代码快* 3.判断共享数据是否到了末尾(到了末尾)* 4.判断共享数据是否到了末尾(没到末尾,执行核心逻辑)*/while(true){synchronized (Desk.lock){if(Desk.count == 0){break;}else {// 判断桌子上是否有食物if(Desk.foodFlag == 1) {// 如果有,就等待try {Desk.lock.wait();} catch (InterruptedException e) {throw new RuntimeException(e);}}else {// 如果没有,就制作食物System.out.println("厨师在制作面条");// 修改桌子上的食物状态Desk.foodFlag = 1;// 叫醒等待的消费者开吃Desk.lock.notifyAll();}}}}}
}
class Foodie extends Thread {@Overridepublic void run() {/*** 1.循环* 2.同步代码快* 3.判断共享数据是否到了末尾(到了末尾)* 4.判断共享数据是否到了末尾(没到末尾,执行核心逻辑)*/while(true){synchronized (Desk.lock) {if(Desk.count == 0){break;}else {// 先判断桌子上面是否有面条if(Desk.foodFlag == 0) {// 如果没有,就等待// 不管是等待还是唤醒都是需要用锁对象进行处理的try {Desk.lock.wait(); // 让当前线程跟锁进行绑定} catch (InterruptedException e) {throw new RuntimeException(e);}}else {// 把吃的总数-1Desk.count --;// 如果有,就开吃System.out.println("吃货在吃面条,还能再吃 " + Desk.count + "碗!");// 吃完之后,就唤醒厨师继续做Desk.lock.notifyAll();// 最后要修改桌子的状态Desk.foodFlag = 0;}}}}}
}class Desk extends Thread {/*** 作用:控制生产者和消费者的执行*/// 是否有面条 0:没有面条 1:有面条public static int foodFlag = 0;// 总个数public static int count = 10;// 锁对象public static Object lock = new Object();
}

2. 使用阻塞队列实现

阻塞队列,当存入数据时,如果空间已满就阻塞;当取出数据时,如果没有数据在队列中就阻塞;

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

public class demo12 {public static void main(String[] args) {/*** 需求:利用阻塞队列完成生产者和消费者* 细节:*      生产者和消费者必须使用同一个阻塞队列*///1. 创建阻塞队列的对象ArrayBlockingQueue<String> queue = new ArrayBlockingQueue<String>(1);//2. 创建线程的对象,并把阻塞队列传递过去Cook12 cook12 = new Cook12(queue);Foodie12 foodie12 = new Foodie12(queue);cook12.start();foodie12.start();}
}
class Cook12 extends Thread {//只定义,不给值ArrayBlockingQueue<String> queue;//让构造方法获取public Cook12(ArrayBlockingQueue<String> queue) {this.queue = queue;}@Overridepublic void run() {while(true){//不断的把面条放到阻塞队列之中try {//ctrl+b进入源码//put底层已经实现了锁,所以无需再写锁queue.put("面条");System.out.println("厨师做了面条");} catch (InterruptedException e) {throw new RuntimeException(e);}}}
}class Foodie12 extends Thread {ArrayBlockingQueue<String> queue;//让构造方法获取public Foodie12(ArrayBlockingQueue<String> queue) {this.queue = queue;}@Overridepublic void run() {while(true){//不断的从阻塞队列中获取面条try {//ctrl+b进入源码//put底层已经实现了锁,所以无需再写锁String take = queue.take();System.out.println(take);} catch (InterruptedException e) {throw new RuntimeException(e);}}}
}

3. 多线程的6种状态

在这里插入图片描述
在这里插入图片描述

http://www.lryc.cn/news/94362.html

相关文章:

  • day2 QT按钮与容器
  • JPA 批量插入较大数据 解决性能慢问题
  • 为啥离不了 linux
  • 基于分形的置乱算法和基于混沌系统的置乱算法哪种更安全?
  • pve使用cloud-image创建ubuntu模板
  • shiro入门
  • 开源 sysgrok — 用于分析、理解和优化系统的人工智能助手
  • Gitlab保护分支与合并请求
  • ad18学习笔记九:输出文件
  • PostgreSQL 内存配置 与 MemoryContext 的生命周期
  • vue3 组件间通信的方式(setup语法糖写法)
  • 【Cache】Rsync远程同步
  • Gitlab升级报错一:rails_migration[gitlab-rails] (gitlab::database_migrations line 51)
  • chatGPT流式回复是怎么实现的
  • 使用SpringEL获得字符串中的表达式运算结果
  • 力扣 39. 组合总和
  • 基于BES系列蓝牙耳机NTC充电电池保护电路设计
  • 13-C++算法笔记-递归
  • 从古代八卦探究计算机的八进制
  • Linux shell mkfs.ext4命令参数使用
  • 【Docker】子系统与其相关名词的界定、Control Groups等详细讲解
  • spring事务的传播性与隔离性
  • 【设计模式】模板方法与策略模式的结合使用
  • Jmeter实现参数加密
  • Solon Web 开发:四、认识请求上下文(Context)
  • docker安装RocketMQ(附填坑经验connect to <172.17.0.3:10909> failed)
  • GRU、LSTM、注意力机制(第八次组会)
  • 问题杂谈(三十六)@RequestBody、@RequestParam和@PathVariable三个注解的区别和使用
  • Flutter学习四:Flutter开发基础(六)调试Flutter应用
  • 新的开始(开始更新笔记)