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();
        }
    }
}
相关推荐
ZenosDoron5 分钟前
keil软件修改字体,Asm editor,和C/C++ editor的区别
c语言·开发语言·c++
qq_6543669821 分钟前
如何排查Oracle客户端连接慢_DNS解析超时与sqlnet配置优化
jvm·数据库·python
焦糖玛奇朵婷23 分钟前
解锁扭蛋机小程序的五大优势
java·大数据·服务器·前端·小程序
山栀shanzhi26 分钟前
C/C++之:构造函数为什么不能设置为虚函数?
开发语言·c++·面试
lsx20240628 分钟前
.toggleClass() 方法详解
开发语言
yuan1999734 分钟前
C&CG(列与约束生成)算法,来解决“风光随机性”下的微网鲁棒配置问题
c语言·开发语言·算法
SamDeepThinking35 分钟前
别让一个超时的第三方http接口拖垮所有接口
java·后端·架构
李白的天不白43 分钟前
读到数据为undefind是的几种情况
开发语言·javascript·ecmascript
YaBingSec1 小时前
玄机靶场:供应链安全-供应链应急-Part2 通关笔记
java·笔记·安全
迷途酱1 小时前
手写一个 AI Agent:从 Function Calling 到自动化任务链
python