编写一个Java程序,其中包含三个线程: 厨师(Chef)、服务员(Waiter)和顾客(Customer)

本文主要是介绍编写一个Java程序,其中包含三个线程: 厨师(Chef)、服务员(Waiter)和顾客(Customer),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

编写一个Java程序,其中包含三个线程: 厨师(Chef)、服务员(Waiter)和顾客(Customer)。他们的行动如下:

  • 厨师准备菜肴,每次准备一个。
  • 服务员等待菜肴准备好,然后将其送到顾客那里。
  • 顾客等待服务员送来菜看后才开始吃。
  • 所有三个角色应该循环进行他们的行为,直到指定数量的菜肴被制作和消费完。
  • 使用wait()和notify0)确保线程间正确的通信和同步。
package com;/*** @program: RestaurantDemo* @description: 餐厅类* @author: Casey Hu* @create: 2023-12-18 16:13**/public class Restaurant {public static void main(String[] args) throws InterruptedException {Chef chef = new Chef();Waiter waiter = new Waiter(chef);Customer customer = new Customer(waiter);chef.start();waiter.start();customer.start();chef.join();waiter.join();customer.join();System.out.println("餐厅关闭.");}static class Chef extends Thread {private final int MAX_FOOD = 10;private int foodCount = 0;private boolean isFinished = false;@Overridepublic void run() {while (!isFinished) {try {// 厨师准备菜肴System.out.println("厨师准备菜肴...");Thread.sleep(1000);// 菜肴准备好后唤醒服务员foodCount++;synchronized (this) {this.notify();}if (foodCount >= MAX_FOOD) {isFinished = true;}} catch (InterruptedException e) {e.printStackTrace();}}}}static class Waiter extends Thread {private Chef chef;public Waiter(Chef chef) {this.chef = chef;}@Overridepublic void run() {while (!chef.isFinished) {try {synchronized (chef) {// 等待厨师准备菜肴chef.wait();}// 将菜肴送到顾客那里System.out.println("服务员将菜肴送到顾客那里...");Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}}}}static class Customer extends Thread {private Waiter waiter;public Customer(Waiter waiter) {this.waiter = waiter;}@Overridepublic void run() {while (!waiter.chef.isFinished) {try {// 等待服务员送来菜肴Thread.sleep(1000);// 开始吃菜肴System.out.println("顾客开始吃菜肴...");// 通知服务员可以送下一道菜了synchronized (waiter.chef) {waiter.chef.notify();}} catch (InterruptedException e) {e.printStackTrace();}}}}
}

上述代码中启动了三个子线程:厨师线程、服务员线程和顾客线程。厨师线程使用foodCount计数器来记录制作的菜肴数量,并使用synchronized关键字确保线程安全。每当它制作好一道菜肴后,它唤醒等待的服务员线程。

服务员线程使用synchronized和wait()方法等待厨师线程准备好菜肴。一旦收到通知,它将菜肴送到顾客那里。

顾客线程使用synchronized关键字和notify()方法通知服务员准备下一道菜肴,并在接收到菜肴后开始吃。

主线程使用join()方法等待所有子线程完成。

这个示例使用了Java中的核心多线程机制(Thread、Runnable、synchronized、wait()和notify()等)来模拟餐厅的运作。

运行结果:

在这里插入图片描述

但是有一段代码

public class Restaurant {private static final int MAX_FOOD = 10;private static final Object lock = new Object();private static int foodCount = 0;public static void main(String[] args) {Chef chef = new Chef();Waiter waiter = new Waiter();Customer customer = new Customer();chef.start();waiter.start();customer.start();}static class Chef extends Thread {@Overridepublic void run() {synchronized (lock) {while (foodCount < MAX_FOOD) {try {// 厨师准备菜肴System.out.println("厨师准备菜肴...");Thread.sleep(1000);// 菜肴准备好后唤醒服务员foodCount++;lock.notify();} catch (InterruptedException e) {e.printStackTrace();}}}}}static class Waiter extends Thread {@Overridepublic void run() {synchronized (lock) {while (foodCount < MAX_FOOD) {try {// 等待菜肴准备好lock.wait();// 将菜肴送到顾客那里System.out.println("服务员将菜肴送到顾客那里...");Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}}}}}static class Customer extends Thread {@Overridepublic void run() {synchronized (lock) {while (foodCount < MAX_FOOD) {try {// 等待菜肴送来lock.wait();// 开始吃菜肴System.out.println("顾客开始吃菜肴...");Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}}}}}
}

运行结果:

在这里插入图片描述
就是厨师准备好菜肴之后,main线程结束
所以才会选择子线程的方式,有大佬知道为啥会出现这样的结果吗?评论区聊聊

这篇关于编写一个Java程序,其中包含三个线程: 厨师(Chef)、服务员(Waiter)和顾客(Customer)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/532044

相关文章

java如何解压zip压缩包

《java如何解压zip压缩包》:本文主要介绍java如何解压zip压缩包问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Java解压zip压缩包实例代码结果如下总结java解压zip压缩包坐在旁边的小伙伴问我怎么用 java 将服务器上的压缩文件解压出来,

SpringBoot中SM2公钥加密、私钥解密的实现示例详解

《SpringBoot中SM2公钥加密、私钥解密的实现示例详解》本文介绍了如何在SpringBoot项目中实现SM2公钥加密和私钥解密的功能,通过使用Hutool库和BouncyCastle依赖,简化... 目录一、前言1、加密信息(示例)2、加密结果(示例)二、实现代码1、yml文件配置2、创建SM2工具

Spring WebFlux 与 WebClient 使用指南及最佳实践

《SpringWebFlux与WebClient使用指南及最佳实践》WebClient是SpringWebFlux模块提供的非阻塞、响应式HTTP客户端,基于ProjectReactor实现,... 目录Spring WebFlux 与 WebClient 使用指南1. WebClient 概述2. 核心依

Spring Boot @RestControllerAdvice全局异常处理最佳实践

《SpringBoot@RestControllerAdvice全局异常处理最佳实践》本文详解SpringBoot中通过@RestControllerAdvice实现全局异常处理,强调代码复用、统... 目录前言一、为什么要使用全局异常处理?二、核心注解解析1. @RestControllerAdvice2

Spring IoC 容器的使用详解(最新整理)

《SpringIoC容器的使用详解(最新整理)》文章介绍了Spring框架中的应用分层思想与IoC容器原理,通过分层解耦业务逻辑、数据访问等模块,IoC容器利用@Component注解管理Bean... 目录1. 应用分层2. IoC 的介绍3. IoC 容器的使用3.1. bean 的存储3.2. 方法注

Spring事务传播机制最佳实践

《Spring事务传播机制最佳实践》Spring的事务传播机制为我们提供了优雅的解决方案,本文将带您深入理解这一机制,掌握不同场景下的最佳实践,感兴趣的朋友一起看看吧... 目录1. 什么是事务传播行为2. Spring支持的七种事务传播行为2.1 REQUIRED(默认)2.2 SUPPORTS2

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Java进程异常故障定位及排查过程

《Java进程异常故障定位及排查过程》:本文主要介绍Java进程异常故障定位及排查过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、故障发现与初步判断1. 监控系统告警2. 日志初步分析二、核心排查工具与步骤1. 进程状态检查2. CPU 飙升问题3. 内存

java中新生代和老生代的关系说明

《java中新生代和老生代的关系说明》:本文主要介绍java中新生代和老生代的关系说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、内存区域划分新生代老年代二、对象生命周期与晋升流程三、新生代与老年代的协作机制1. 跨代引用处理2. 动态年龄判定3. 空间分

Java设计模式---迭代器模式(Iterator)解读

《Java设计模式---迭代器模式(Iterator)解读》:本文主要介绍Java设计模式---迭代器模式(Iterator),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录1、迭代器(Iterator)1.1、结构1.2、常用方法1.3、本质1、解耦集合与遍历逻辑2、统一