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();
        }
    }
}
相关推荐
用户83562907805114 小时前
无需 Office:Python 批量转换 PPT 为图片
后端·python
日月云棠15 小时前
各版本JDK对比:JDK 25 特性详解
java
markfeng816 小时前
Python+Django+H5+MySQL项目搭建
python·django
用户83071968408216 小时前
Spring Boot 项目中日期处理的最佳实践
java·spring boot
JavaGuide16 小时前
Claude Opus 4.6 真的用不起了!我换成了国产 M2.5,实测真香!!
java·spring·ai·claude code
GinoWi16 小时前
Chapter 2 - Python中的变量和简单的数据类型
python
IT探险家17 小时前
Java 基本数据类型:8 种原始类型 + 数组 + 6 个新手必踩的坑
java
JordanHaidee17 小时前
Python 中 `if x:` 到底在判断什么?
后端·python
花花无缺17 小时前
搞懂new 关键字(构造函数)和 .builder() 模式(建造者模式)创建对象
java
用户9083246027317 小时前
Spring Boot + MyBatis-Plus 多租户实战:从数据隔离到权限控制的完整方案
java·后端