编写一个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流操作对文件的分割和合并

学习文件的输入输出流,自己做一个小的示例,对文件进行分割和合并。     下面是代码: <span style="font-size:14px;">package com.dufy.file;import java.io.File;import java.io.FileInputStream;import java.io.FileOutputStream;import

java 通过Ajax前台传参数 并用 HttpURLConnection Post方式访问对外的接口

前两天做项目遇到一个问题,就是在自己的项目中要去访问项目外部的接口,从自己的项目中传参数过去,通过调用 对方提供的接口去获取想要得到的数据!第一次接触到在自己项目中去访问和调用外部的资源,然后在网上去找资料,看有没有相关的资料可以参考,然后通过参考其他人的博客资料,最终把这个问题解决了。自己总结一下这个过程,也供遇到相同或者类似问题的朋友可以快速的定位和解决问题。      下面讲一下我

javaweb学习-jstl-c:forEach中 varStatus的属性简介

varStatus是<c:forEach>jstl循环标签的一个属性,varStatus属性。就拿varStatus=“status”来说,事实上定义了一个status名的对象作为varStatus的绑定值。该绑定值也就是status封装了当前遍历的状态,比如,可以从该对象上查看是遍历到了第几个元素:${status.count} 我们常会用c标签来遍历需要的数据,为了方便使用,varSta

javaweb—jstl如何循环List中的Map数据

第一种方式: 1:后台代码(测试) List<Map<String, Object>> list = new ArrayList<Map<String,Object>>();Map<String, Object> map = null;for (int i = 0; i < 4; i++) {map = new HashMap<String, Object>();map.put("id",

日志记录—Java中的日志——Java.util.logging、log4j、commons-logging

转载地址  http://blog.csdn.net/luoweifu/article/details/46495045 Java中给项目程序添加log主要有三种方式,一使用JDK中的java.util.logging包,一种是log4j,一种是commons-logging。其中log4j和commons-logging都是apache软件基金会的开源项目。这三种方式的区别如下: J

Spring AOP 实现监控方法执行的时间(统计service中方法执行的时间)

项目中有时候会遇到统计方法执行的时间,来对项目进行优化!下面是我自己在工作中遇到的问题,和我自己的解决方法。 要统计出项目中方法执行时间大于1秒的那些方法!我们的项目开发使用的是SpringMVC 那么首先想到使用 Aop Aspet 切面统计,那样子更加方便也高效。 1:打开切面!因为项目使用的SpringMVC,项目中的配置文件就配置的 <aop:aspectj-autoproxy pro

java学习—null和isEmpty 区别

String fly1 = new String(); String fly2 = ""; String fly3 = null; 解释如下: 此时fly1是分配了内存空间,但值为空,是绝对的空,是一种有值(值存在为空而已) 此时fly2是分配了内存空间,值为空字符串,是相对的空,是一种有值(值存在为空字串) 此时fly3是未分配内存空间,无值,是一种无值(

javaWeb学习—getRequestURI,getRequestURL等的学习

我使用的是SpringMVC框架,做一个小的例子,说明一下对这个内容的学习和理解! 1:我的项目名称为 dufyun_SpringMVC  2:我测试的地址为 ${pageContext.servletContext.contextPath}/testName 3;后台获取的代码为: @RequestMapping(value="/testName",method=Req

java基础—java中使用final关键字的总结

有时候我,们希望某些东西是亘古不变的,可以使用final关键字完成这个重任! final学习总结: 1:final + 属性 如果属性是基本数据类型(byte 字节型short 短整型int 普通整型char 字符型float 浮点型long 长整型double 双精度),则变为常量,其值不能被更改;如果属性是引用类型,则引用地址不能被更改。(final 修饰一个对象,那么这个对象的引用地址

java多线程—java线程的创建和线程的生命周期

<span style="font-size:14px;">package com.dufy.thread;/***1 线程学习* @author aflyun * */public class TestCreateThread {public static void main(String[] args) {testThread tt = new testThread();testRunab