Java多线程间的通信:生产者消费者问题

逻辑分析



代码实现

java 复制代码
package ThreadCommunction;

import sun.security.krb5.internal.crypto.Des;

import java.util.Date;

//目标:了解线程通信
public class ThreadTest {
    public static void main(String[] args) {
        //需求:3个生产者线程,负责产包子,每个线程每次只能生产1个包子放在桌子上
        //     2个消费者线程复杂吃包子,每人每次只能从桌子上拿1个包子吃
        Desk desk = new Desk();

        //创建3个生产者线程(3个厨师)
        new Thread(()-> {
            while (true) {
                desk.put();
            }

        },"厨师1").start();


        new Thread(()-> {
            while (true) {
                desk.put();
            }

        },"厨师2").start();


        new Thread(()-> {
            while (true) {
                desk.put();
            }

        },"厨师3").start();

        //创建2个消费者线程(2个吃货)
        new Thread(()-> {
            while (true) {
                desk.get();
            }

        },"吃货1").start();


        new Thread(()-> {
            while (true) {
                desk.get();
            }

        },"吃货2").start();

    }
}
java 复制代码
package ThreadCommunction;

import java.util.ArrayList;
import java.util.List;

public class Desk {
    private List<String> list = new ArrayList<>();

    //放一个包子的方法
    //厨师1、厨师2、厨师3
    public synchronized void put() {
        try {
            String name =  Thread.currentThread().getName();
            //判断是否有包子
            if(list.size() ==0){
                list.add(name + "做的肉包子");
                System.out.println(name + "做了一个肉包子~~");
                Thread.sleep(2000);

                //唤醒别人,等待自己(顺序不能混)
                this.notify();
                this.wait();
            }else {
                //有包子了,不做了
                //唤醒别人,等待自己
                this.notify();
                this.wait();
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    //取一个包子
    //吃货1、吃货2
    public synchronized void get() {
        try {
            String name = Thread.currentThread().getName();
            if (list.size() == 1){
                //有包子,吃了
                System.out.println(name+"吃了"+list.get(0));
                //清空
                list.clear();
                //唤醒别人,等待自己
                Thread.sleep(1000);
                this.notify();
                this.wait();
            }else {
                //没有包子  唤醒别人,等待自己
                this.notify();
                this.wait();
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}
相关推荐
OperateCode几秒前
AutoVideoMerge:让二刷更沉浸的自动化视频处理脚本工具
python·opencv·ffmpeg
蔡俊锋1 分钟前
Javar如何用RabbitMQ订单超时处理
java·python·rabbitmq·ruby
跟橙姐学代码9 分钟前
学Python别死记硬背,这份“编程生活化笔记”让你少走三年弯路
前端·python
天天摸鱼的java工程师18 分钟前
Snowflake 雪花算法优缺点(Java老司机实战总结)
java·后端·面试
Miraitowa_cheems1 小时前
LeetCode算法日记 - Day 11: 寻找峰值、山脉数组的峰顶索引
java·算法·leetcode
海梨花1 小时前
【从零开始学习Redis】项目实战-黑马点评D2
java·数据库·redis·后端·缓存
共享家95271 小时前
linux-高级IO(上)
java·linux·服务器
Sammyyyyy1 小时前
2025年,Javascript后端应该用 Bun、Node.js 还是 Deno?
开发语言·javascript·node.js
橘子郡1231 小时前
观察者模式和发布订阅模式对比,Java示例
java
指针满天飞1 小时前
Collections.synchronizedList是如何将List变为线程安全的
java·数据结构·list