Java多线程卖包子问题(附多种实现方式)

news2024/9/23 13:29:01

前提条件:生产者生产包子,但是包子总数不得超过10个,消费者有包子就购买

第一种:通过wait()和notify()来实现

在这个实例中,生产者和消费者通过wait()和notify()来进行通讯,再通过synchronized 块来确保线程安全,当包子数量达到最大容量时生产者线程会进入等待状态,直到有消费者购买包子释放空间。当包子数量为 0 时消费者线程会进入等待状态,直到有生产者生产包子。

public class ThreadTest {

    public static void main(String[] args) {
        Bakery bakery = new Bakery();

        Thread producerThread = new Thread(() -> {
            try {
                while (true) {
                    bakery.produce();
                    Thread.sleep(1000); // 模拟生产时间
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });

        Thread consumerThread = new Thread(() -> {
            try {
                while (true) {
                    bakery.consume();
                    Thread.sleep(2000); // 模拟购买时间
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });

        producerThread.start();
        consumerThread.start();
    }
}

class Bakery {
    private int numOfBuns = 0;
    private final int capacity = 10;

    public synchronized void produce() throws InterruptedException {
        while (numOfBuns == capacity) {
            wait(); // 等待直到有消费者买走包子,释放空间
        }
        numOfBuns++;
        System.out.println("包子数量+++++++++++++++++" + numOfBuns);
        notify(); // 通知等待的消费者可以买包子了
    }

    public synchronized void consume() throws InterruptedException {
        while (numOfBuns == 0) {
            wait(); // 等待直到有包子可买
        }
        numOfBuns--;
        System.out.println("包子数量—————————————————" + numOfBuns);
        notify(); // 通知等待的生产者可以继续生产包子了
    }
}

运行一下
可以看到因为生产速度比购买速度快,包子最后一直维持在10个

在这里插入图片描述

第二种:通过BlockingQueue来实现

在这个示例中,BlockingQueue 可以自动处理线程安全和阻塞的问题,生产者调用 put() 方法向队列中放入包子,如果队列已满则会阻塞。消费者调用 take() 方法从队列中取出包子,如果队列为空则会阻塞。
这种方式简化了代码,提高了可读性和可维护性。

import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.BlockingQueue;

public class ThreadTest {

    public static void main(String[] args) {
        Bakery bakery = new Bakery();

        Thread producerThread = new Thread(() -> {
            try {
                while (true) {
                    bakery.produce();
                    Thread.sleep(1000); // 模拟生产时间
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });

        Thread consumerThread = new Thread(() -> {
            try {
                while (true) {
                    bakery.consume();
                    Thread.sleep(2000); // 模拟购买时间
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });

        producerThread.start();
        consumerThread.start();
    }
}

class Bakery {
    private BlockingQueue<Integer> queue = new ArrayBlockingQueue<>(10);

    public void produce() throws InterruptedException {
        queue.put(1); // 生产一个包子,如果队列已满,则阻塞
        System.out.println("包子数量+++++++++++++++++" + queue.size());
    }

    public void consume() throws InterruptedException {
        queue.take(); // 消费一个包子,如果队列为空,则阻塞
        System.out.println("包子数量—————————————————" + queue.size());
    }
}

运行一下
可以发现第一个数据有些奇怪,这是因为生产者生产了一个包子放入队列马上就被消费者取走了,这个过程发生在System.out.println之前,所以打印出来的包子数量变成了0,最后包子数量还是维持在10个,然后买一个生产一个

在这里插入图片描述

第三种:通过 Object 类型的锁对象来实现

在这个示例中,使用一个 Object 类型的锁对象 lock 来进行同步。生产者在生产包子时获取 lock 锁,如果缓冲区已满,则调用 wait() 方法进入等待状态,直到有消费者购买了包子并释放了锁。消费者在购买包子时也获取 lock 锁,如果缓冲区为空,则调用 wait() 方法进入等待状态,直到有生产者生产了包子并释放了锁。当生产者生产了包子或消费者购买了包子后,调用 notify() 方法通知等待中的线程。

public class ThreadTest {

    public static void main(String[] args) {
        Bakery bakery = new Bakery();

        Thread producerThread = new Thread(() -> {
            try {
                while (true) {
                    bakery.produce();
                    Thread.sleep(1000); // 模拟生产时间
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });

        Thread consumerThread = new Thread(() -> {
            try {
                while (true) {
                    bakery.consume();
                    Thread.sleep(2000); // 模拟购买时间
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });

        producerThread.start();
        consumerThread.start();
    }
}

class Bakery {
    private final Object lock = new Object();
    private int numOfBuns = 0;
    private final int capacity = 10;

    public void produce() throws InterruptedException {
        synchronized (lock) {
            while (numOfBuns == capacity) {
                lock.wait(); // 缓冲区已满,生产者等待
            }
            numOfBuns++;
            System.out.println("包子数量+++++++++++++++++" + numOfBuns);
            lock.notify(); // 通知消费者可以购买包子
        }
    }

    public void consume() throws InterruptedException {
        synchronized (lock) {
            while (numOfBuns == 0) {
                lock.wait(); // 缓冲区为空,消费者等待
            }
            numOfBuns--;
            System.out.println("包子数量—————————————————" + numOfBuns);
            lock.notify(); // 通知生产者可以继续生产包子
        }
    }
}

运行一下
数据和第一种方式差不多

在这里插入图片描述

第四种:通过Semaphore来实现

在这个示例中,使用信号量Semaphore来控制包子的数量,初始设置了10个许可证。生产者在生产包子时先获取 spaceAvailable的许可证,然后生产一个包子,最多可以获取10个,消费者在购买包子时先释放 spaceAvailable的许可证,然后消费一个包子。

import java.util.concurrent.Semaphore;

public class ThreadTest {

    public static void main(String[] args) {
        Bakery bakery = new Bakery();

        Thread producerThread = new Thread(() -> {
            try {
                while (true) {
                    bakery.produce();
                    Thread.sleep(1000); // 模拟生产时间
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });

        Thread consumerThread = new Thread(() -> {
            try {
                while (true) {
                    bakery.consume();
                    Thread.sleep(2000); // 模拟购买时间
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });

        producerThread.start();
        consumerThread.start();
    }
}

class Bakery {
    private Semaphore spaceAvailable = new Semaphore(10); // 初始为 10,表示有 10 个空位
    private int numOfBuns = 0;

    public void produce() throws InterruptedException {
        spaceAvailable.acquire(); // 获取一个空位的信号量
        numOfBuns++;
        System.out.println("包子数量+++++++++++++++++" + numOfBuns);
    }

    public void consume() throws InterruptedException {
        spaceAvailable.release(); // 释放一个空位的信号量
        numOfBuns--;
        System.out.println("包子数量—————————————————" + numOfBuns);
    }
}

运行一下
和第一种方式差不多

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1490356.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Java中文件的相关知识及文件IO操作

在我们日常生活中&#xff0c;会把许多东西都称之为文件。比如&#xff0c;一份纸质报告&#xff0c;或u盘中的一些文档&#xff0c;都会把它们称为文件。那么&#xff0c;这里说的文件是以操作系统的角度出发的。在操作系统中&#xff0c;会把许多硬件设备和软件资源都抽象成“…

【STM32】STM32学习笔记-读写内部FLASH 读取芯片ID(49)

00. 目录 文章目录 00. 目录01. FLASH概述02. 读写内部FLASH接线图03. 读写内部FLASH相关API04. 读写内部FLASH程序示例05. 读写芯片ID接线图06. 读写芯片ID程序示例07. 程序示例下载08. 附录 01. FLASH概述 STM32F10xxx内嵌的闪存存储器可以用于在线编程(ICP)或在程序中编程(…

yolov9训练

目录 说明 1、下载代码安装新的python环境 2、准备数据 3、修改代码 说明 本文参考该博主的文章&#xff0c;在已经有数据的情况&#xff0c;进行简单总结。需要详细版见原文链接如下&#xff1a;YOLOV9保姆级教程-CSDN博客 1、下载代码安装新的python环境 代码下载&…

【数据结构和算法初阶(C语言)】带环链表问题详解(快慢指针的烧脑应用)

目录 1.铺垫-----带环链表基本了解 2. 题目&#xff1a;环形链表 3.环形链表|| ​编辑 3.1题解1 3.2 题解2 4.总结 1.铺垫-----带环链表基本了解 环形链表题目启迪&#xff1a; 环形链表特点&#xff1a;遍历链表会出现一模一样的地址 2. 题目&#xff1a;环形链表 给…

如何使用 ArcGIS Pro 制作三维地形图

伴随硬件性能的提高和软件算法的优化&#xff0c;三维地图的应用场景会越来越多&#xff0c;这里为大家介绍一下在ArcGIS Pro怎么制作三维地形图&#xff0c;希望能对你有所帮助。 数据来源 教程所使用的数据是从水经微图中下载的DEM和影像数据&#xff0c;除了DEM和影像数据…

笨办法学 Python3 第五版(预览)(二)

原文&#xff1a;Learn Python the Hard Way, 5th Edition (Early Release) 译者&#xff1a;飞龙 协议&#xff1a;CC BY-NC-SA 4.0 练习 19&#xff1a;函数和变量 现在你将把函数与你从之前练习中了解到的变量结合起来。如你所知&#xff0c;变量给数据片段一个名称&#x…

Spring Boot项目中不使用@RequestMapping相关注解,如何动态发布自定义URL路径

一、前言 在Spring Boot项目开发过程中&#xff0c;对于接口API发布URL访问路径&#xff0c;一般都是在类上标识RestController或者Controller注解&#xff0c;然后在方法上标识RequestMapping相关注解&#xff0c;比如&#xff1a;PostMapping、GetMapping注解&#xff0c;通…

⭐每天一道leetcode:21.合并两个有序链表(简单;双指针)

⭐今日份题目 将两个升序链表合并为一个新的 升序 链表并返回。新链表是通过拼接给定的两个链表的所有节点组成的。 示例1 输入&#xff1a;l1 [1,2,4], l2 [1,3,4] 输出&#xff1a;[1,1,2,3,4,4] 示例2 输入&#xff1a;l1 [], l2 [] 输出&#xff1a;[] 示例3 输入…

【机器学习】包裹式特征选择之递归特征添加法

&#x1f388;个人主页&#xff1a;豌豆射手^ &#x1f389;欢迎 &#x1f44d;点赞✍评论⭐收藏 &#x1f917;收录专栏&#xff1a;机器学习 &#x1f91d;希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出指正&#xff0c;让我们共同学习、交流进…

Unity 摄像机的深度切换与摄像机画面投影

摄像机可选&#xff1a;透视、正交 正交类似投影&#xff0c;1比1 透视类似人眼&#xff0c;近大远小 摄像机投影 在项目中新建&#xff1a;渲染器纹理 将新建纹理拖动到相机的目标纹理中 新建一个平面&#xff0c;将新建材质组件放到平面中即可。 相机深度切换 使用代…

探索Java开发面试笔记:以听为目的,助力编程技术提升与面试准备

文章目录 一、制作背景介绍二、 Java开发面试笔记&#xff1a;为你的编程之路加速2.1 公众号主题和目标读者群体2.2 为什么面试笔记对于提高编程视野和技术至关重要2.3 親測效率 三、形式案例3.1 文章形式3.2 手机案例3.3 电脑案例 一、制作背景介绍 做公众号的背景&#xff1a…

十九 超级数据查看器 讲解稿 分栏功能

十九 超级数据查看器 讲解稿 分栏功能 点击此处观看视频教程 讲解稿全文: 大家好&#xff0c;这讲介绍一下 &#xff0c;超级数据查看器的分栏功能。 分栏功能设计的初衷是为了让用户同时同地查询两个表格的数据&#xff0c;方便比较&#xff0c;获得更清晰的查询结果 分栏功…

模型优化_XGBOOST学习曲线及改进,泛化误差

代码 from xgboost import XGBRegressor as XGBR from sklearn.ensemble import RandomForestRegressor as RFR from sklearn.linear_model import LinearRegression as LR from sklearn.datasets import load_boston from sklearn.model_selection import train_test_split,c…

如何添加极狐GitLab Runner 信任域名证书

本文作者 徐晓伟 极狐Gitlab Runner 信任实例域名证书&#xff0c;用于注册注册极狐 GitLab Runner。 问题 参见 极狐gitlab-runner-host.md 说明 解决方案是使用颁发给域名 gitlab.test.helm.xuxiaowei.cn 的证书&#xff0c;可以使用自己的域名去各大云厂商免费申请&#…

重学SpringBoot3-yaml文件配置

重学SpringBoot3-yaml文件配置 引言YAML 基本语法YAML 数据类型YAML 对象YAML 数组复合结构标量引用 YAML 文件结构Spring Boot 中的 YAML 配置注意事项总结参考 引言 YAML&#xff08;YAML Ain’t Markup Language&#xff09;是一种常用于配置文件的数据序列化格式&#xff…

Unity3D

一、C# 输入输出 二、三维数学

CSS变量和@property

CSS变量 var() CSS 变量是由CSS作者定义的实体&#xff0c;其中包含要在整个文档中重复使用的特定值。使用自定义属性来设置变量名&#xff0c;并使用特定的 var() 来访问。&#xff08;比如 color: var(--main-color);&#xff09;。 基本用法 CSS变量定义的作用域只在定义该…

搞定国科金 必备王炸新技术!凌恩生物重磅推出微生物单细胞测序产品

单细胞异质性研究如火如荼&#xff0c;但原核生物研究却是个“坎”。 现有常规的原核生物研究&#xff0c;都集中于单菌群落或微生态大群体&#xff0c;只能从宏观角度研究群体状态&#xff0c;而经典的单细胞RNA测序技术无法应用于细菌。 单细胞技术应用于原核生物的几点障碍…

window10 安装配置docker

前言&#xff08;重要&#xff09;&#xff1a;确认window10版本已经更新到最新版 随着时间推移&#xff0c;docker对window版本的支持也在变&#xff0c;截至2024年3月份&#xff0c;支持win10最低版本号&#xff1a;22H2,操作系统最低版本&#xff1a;19045.2965&#xff0c…

学编程怎么样才能更快入手,编程怎么简单易学

学编程怎么样才能更快入手&#xff0c;编程怎么简单易学 一、前言 对于初学编程建议先从简单入手&#xff0c;然后再学习其他复杂的编程语言。 今天给大家分享的中文编程开发语言工具 进度条构件的用法。 编程入门视频教程链接 https://edu.csdn.net/course/detail/39036 …