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();
        }
    }
}
相关推荐
摇滚侠几秒前
Maven 入门+高深 单一架构案例 54-59
java·架构·maven·intellij-idea
VidDown14 分钟前
Webhook 调试器:让第三方回调“原形毕露”
java·开发语言·javascript·编辑器·postman
折哥的程序人生 · 物流技术专研26 分钟前
Java 23 种设计模式:从踩坑到精通 | 原型模式 —— 克隆对象,深拷贝与浅拷贝的坑你踩过吗?
java·设计模式·架构·原型模式·单一职责原则
装不满的克莱因瓶38 分钟前
基于 OpenResty 扩展开发实现动态服务注册与发现能力
java·开发语言·架构·openresty
程序员小羊!1 小时前
06Java 异常机制与常用类
java
颜酱1 小时前
LangChain 输出解析器:把模型回复变成你要的数据
python·langchain
2401_873479401 小时前
企业安全运营中,如何用IP离线库提前发现失陷主机?三步实现风险画像
网络·数据库·python·tcp/ip·ip
weixin_523185321 小时前
Java基础知识总结(四):引用数据类型与参数传递机制
java·开发语言·python
Nayxxu1 小时前
Claude API 生产稳定性设计:超时、降级、备用模型和告警怎么做
开发语言·php
王cb1 小时前
WinRT Server and Client c#
开发语言·c#