Blog PTA 4-5-期中
菜单计价程序代码分析---4-5
7-1 菜单计价程序-4
输入结构
按输入顺序输出每一桌的订单记录处理信息,包括:
桌号标识格式:table + 序号 +英文空格+ 日期(格式:YYYY/MM/DD)+英文空格+ 时间(24小时制格式: HH/MM/SS)
菜品记录格式:
菜名+英文空格+基础价格
如果有多条相同的菜名的记录,菜品的基础价格以最后一条记录为准。
点菜记录格式:序号+英文空格+菜名+英文空格+份额+英文空格+份数注:份额可输入(1/2/3), 1代表小份,2代表中份,3代表大份。
删除记录格式:序号 +英文空格+delete
代点菜信息包含:桌号+英文空格+序号+英文空格+菜品名称+英文空格+份额+英文空格+分数
最后一条记录以“end”结束。
输出结构
1、桌号,格式:table+英文空格+桌号+”:”+英文空格
2、按顺序输出当前这一桌每条订单记录的处理信息,
每条点菜记录输出:序号+英文空格+菜名+英文空格+价格。其中的价格等于对应记录的菜品*份数,序号是之前输入的订单记录的序号。如果订单中包含不能识别的菜名,则输出“** does not exist”,**是不能识别的菜名
如果删除记录的序号不存在,则输出“delete error”
最后按输入顺序一次输出每一桌所有菜品的总价(整数数值)格式:table+英文空格+桌号+“:”+英文空格+当前桌的原始总价+英文空格+当前桌的计算折扣后总价
数据处理
设计点菜计价程序,根据输入的信息,计算并输出总价格。
输入内容按先后顺序包括两部分:菜单、订单,最后以"end"结束。
菜单由一条或多条菜品记录组成,每条记录一行
每条菜品记录包含:菜名、基础价格 两个信息。
订单分:桌号标识、点菜记录和删除信息、代点菜信息。每一类信息都可包含一条或多条记录,每条记录一行或多行。
桌号标识独占一行,包含两个信息:桌号、时间。
桌号以下的所有记录都是本桌的记录,直至下一个桌号标识。
点菜记录包含:序号、菜名、份额、份数。份额可选项包括:1、2、3,分别代表小、中、大份。
不同份额菜价的计算方法:小份菜的价格=菜品的基础价格。中份菜的价格=菜品的基础价格1.5。小份菜的价格=菜品的基础价格2。如果计算出现小数,按四舍五入的规则进行处理。
删除记录格式:序号 delete
标识删除对应序号的那条点菜记录。
如果序号不对,输出"delete error"
代点菜信息包含:桌号 序号 菜品名称 份额 分数
代点菜是当前桌为另外一桌点菜,信息中的桌号是另一桌的桌号,带点菜的价格计算在当前这一桌。
程序最后按输入的桌号从小到大的顺序依次输出每一桌的总价(注意:由于有代点菜的功能,总价不一定等于当前桌上的菜的价格之和)。
每桌的总价等于那一桌所有菜的价格之和乘以折扣。如存在小数,按四舍五入规则计算,保留整数。
折扣的计算方法(注:以下时间段均按闭区间计算):
周一至周五营业时间与折扣:晚上(17:00-20:30)8折,周一至周五中午(10:30--14:30)6折,其余时间不营业。
周末全价,营业时间:9:30-21:30
如果下单时间不在营业范围内,输出"table " + t.tableNum + " out of opening hours"
注意事项
1、菜谱信息与订单信息混合,应忽略夹在订单信息中的菜谱信息。输出:"invalid dish"
2、桌号所带时间格式合法(格式见输入格式部分说明,其中年必须是4位数字,月、日、时、分、秒可以是1位或2位数),数据非法,比如:2023/15/16 ,输出桌号+" date error"
3、同一桌菜名、份额相同的点菜记录要合并成一条进行计算,否则可能会出现四舍五入的误差。
4、重复删除,重复的删除记录输出"deduplication :"+序号。
5、代点菜时,桌号不存在,输出"Table number :"+被点菜桌号+" does not exist";本次作业不考虑两桌记录时间不匹配的情况。
6、菜谱信息中出现重复的菜品名,以最后一条记录为准。
7、如果有重复的桌号信息,如果两条信息的时间不在同一时间段,(时段的认定:周一到周五的中午或晚上是同一时段,或者周末时间间隔1小时(不含一小时整,精确到秒)以内算统一时段),此时输出结果按不同的记录分别计价。
8、重复的桌号信息如果两条信息的时间在同一时间段,此时输出结果时合并点菜记录统一计价。前提:两个的桌号信息的时间都在有效时间段以内。计算每一桌总价要先合并符合本条件的饭桌的点菜记录,统一计价输出。
9、份额超出范围(1、2、3)输出:序号+" portion out of range "+份额,份额不能超过1位,否则为非法格式,参照第13条输出。
10、份数超出范围,每桌不超过15份,超出范围输出:序号+" num out of range "+份数。份数必须为数值,最高位不能为0,否则按非法格式参照第16条输出。
11、桌号超出范围[1,55]。输出:桌号 +" table num out of range",桌号必须为1位或多位数值,最高位不能为0,否则按非法格式参照第16条输出。
12、菜谱信息中菜价超出范围(区间(0,300)),输出:菜品名+" price out of range "+价格,菜价必须为数值,最高位不能为0,否则按非法格式参照第16条输出。
13、时间输入有效但超出范围[2022.1.1-2023.12.31],输出:"not a valid time period"
14、一条点菜记录中若格式正确,但数据出现问题,如:菜名不存在、份额超出范围、份数超出范围,按记录中从左到右的次序优先级由高到低,输出时只提示优先级最高的那个错误。
15、每桌的点菜记录的序号必须按从小到大的顺序排列(可以不连续,也可以不从1开始),未按序排列序号的输出:"record serial number sequence error"。当前记录忽略。(代点菜信息的序号除外)
16、所有记录其它非法格式输入,统一输出"wrong format"
17、如果记录以“table”开头,对应记录的格式或者数据不符合桌号的要求,那一桌下面定义的所有信息无论正确或错误均忽略,不做处理。如果记录不是以“table”开头,比如“tab le 55 2023/3/2 12/00/00”,该条记录认为是错误记录,后面所有的信息并入上一桌一起计算。
题目提供的参考类
菜品类:对应菜谱上一道菜的信息。
Dish {
String name;//菜品名称
int unit_price; //单价
int getPrice(int portion)//计算菜品价格的方法,输入参数是点菜的份额(输入数据只能是1/2/3,代表小/中/大份) }
菜谱类:对应菜谱,包含饭店提供的所有菜的信息。
Menu {
Dish[] dishs ;//菜品数组,保存所有菜品信息
Dish searthDish(String dishName)//根据菜名在菜谱中查找菜品信息,返回Dish对象。
Dish addDish(String dishName,int unit_price)//添加一道菜品信息
}
点菜记录类:保存订单上的一道菜品记录
Record {
int orderNum;//序号
Dish d;//菜品\\
int portion;//份额(1/2/3代表小/中/大份)
int getPrice()//计价,计算本条记录的价格
}
订单类:保存用户点的所有菜的信息。
Order {
Record[] records;//保存订单上每一道的记录
int getTotalPrice()//计算订单的总价
Record addARecord(int orderNum,String dishName,int portion,int num)//添加一条菜品信息到订单中。
delARecordByOrderNum(int orderNum)//根据序号删除一条记录
findRecordByNum(int orderNum)//根据序号查找一条记录
}
特殊的输入格式
菜单输入时增加特色菜,特色菜的输入格式:菜品名+英文空格+基础价格+"T"
例如:麻婆豆腐 9 T
菜价的计算方法:
周一至周五 7折, 周末全价。
注意:不同的四舍五入顺序可能会造成误差,请按以下步骤累计一桌菜的菜价:
计算每条记录的菜价:将每份菜的单价按份额进行四舍五入运算后,乘以份数计算多份的价格,然后乘以折扣,再进行四舍五入,得到本条记录的最终支付价格。
最后将所有记录的菜价累加得到整桌菜的价格。
难度
*****
SourceMonitor

代码如下
import java.util.Scanner; import java.util.ArrayList; import java.util.List; import java.util.Calendar; import java.util.Date; public class Main { public static void main(String[] args) { Work work = new Work(); work.play(); } } class Work { boolean Stop = false; //用来确定订单和菜单混杂在一起的情况 Menu menu; //单位菜单 Record record; //单位记录 Table table; //单位座位 ArrayList<Table> tables = new ArrayList<>(); //座位数组 Work() { //初始化,用来创建基本的类数组 //只能用一个菜单,别单位化 //对所有的table都开放 menu = new Menu(); //单位化,订单? Record应该是必须得单位化 record = new Record(); //定义一个存储Table的一个ArrayList —— tables的动态数组 tables = new ArrayList<>(); } public void play() { //play 菜单程序主要运行的程序 Scanner in = new Scanner(System.in); while (true) { String Line = in.nextLine(); if (Line.equals("end")) { break; } String[] LineArray = Line.split(" "); if (LineArray.length == 2) { //一个空格 if (LineArray[1].equals("delete")) { //删除订单 int num = Integer.parseInt(LineArray[0]); if (table.deletes.contains(num)) { System.out.println("deduplication " + num); continue; } else { Record pp =table.order.find(num); table.order.delARecordByOrderNum(num); table.deletes.add(num); } } else { //菜单添加 if (!Stop) { //菜单还没有输出完 try { int x = Integer.parseInt(LineArray[1]); } catch (Exception e1) { System.out.println("wrong format"); continue; } Dish dish = new Dish(LineArray[0],Integer.parseInt(LineArray[1])); menu.addDish(dish); } else { System.out.println("invalid dish"); } } //continue; } else if (LineArray.length == 3) { if (LineArray[2].equals("T")) { //特色菜 Dish dish = new Dish(LineArray[0],Integer.parseInt(LineArray[1]),LineArray[2]); menu.addDish(dish); } } else if (LineArray.length == 4) { //三个空格,四个数据 if (LineArray[0].equals("table")) { //座号 try { table = new Table(LineArray[1], LineArray[2], LineArray[3],menu); if (table.time_error) { int yy = Integer.parseInt(LineArray[2]); //用来故意报错 } table.Discount(); //设定该table的折扣 System.out.println("table " + table.num + ": "); tables.add(table); Stop = true; } catch (Exception e2) { System.out.println("wrong format"); break; } } else { //添加订单 int number = Integer.parseInt(LineArray[0]); int portion = Integer.parseInt(LineArray[2]); int num = Integer.parseInt(LineArray[3]); if ((3<portion && portion<10)|| portion<1) { System.out.println(number + " portion out of range " + portion); if (num>15) { System.out.println(number + " num out of range " + num); } continue; } if (portion>=10) { System.out.println("not a valid time period"); if (num>15) { System.out.println(number + " num out of range " + num); } continue; } if (num>15) { System.out.println(number + " num out of range " + num); continue; } record = new Record(number, LineArray[1], portion, num); //初始化单位record的序号,菜名,份额,数量 if (menu.searchDish(LineArray[1]) != null) { //有该菜 Record oop = table.order.find_ByName(LineArray[1]); //查看Order中是否有该菜名的Record if (oop!=null && oop.portion == portion) { //如果有该菜名的Record并且portion相等 oop.num += num; continue; } record.d = menu.searchDish(LineArray[1]); //获得该菜名的Dish类 table.order.addARecord(record); //答案在if的前面??? int under_number = table.order.records.size()-1; //存储最后一个的table的下标 //往record中添加菜品,提供单价 //在record里面添加菜品的单价和名字 table.order.records.get(under_number).d = menu.searchDish(LineArray[1]); //获取record的单个总价 int price = table.order.records.get(under_number).getPrice(); //输出单个record的表出 System.out.println(table.order.records.get(under_number).orderNum + " " + LineArray[1] + " " +price); } } //continue; } else if (LineArray.length == 5) { //代点菜 try { int number = Integer.parseInt(LineArray[1]); int portion = Integer.parseInt(LineArray[3]); int num = Integer.parseInt(LineArray[4]); record = new Record(number, LineArray[2], portion, num); if (menu.searchDish(LineArray[2])!=null) { //菜单内有该菜品 //添加record table.order.addARecord(record); int under_number = table.order.records.size()-1; //设定record的菜品单价 table.order.records.get(under_number).d = menu.searchDish(LineArray[2]); int price = table.order.records.get(under_number).getPrice(); System.out.println(number + " table " + table.num + " pay for table " + LineArray[0] + price); } } catch (Exception e2){ System.out.println("wrong format"); break; } } } for (int i = 0 ; i < tables.size() ; i++) { tables.get(i).GetToTalPrice(); } } } // 菜品类:对应菜谱上一道菜的信息。 class Dish { //封装 Dish() { } //空构造器 Dish(String dishName) { this.name = dishName; } Dish(String dishName , int unit_price) { this.name = dishName; this.unit_price = unit_price; } //普通菜的构造器 Dish(String dishName , int unit_price , String pp) { this.name = dishName; this.unit_price = unit_price; if (pp.equals("T")) { TT = true; } } //特色菜的构造器 private String name;//菜品名称 private int unit_price; //单价 private boolean TT = false; //特色菜标志 public int getPrice(int portion) { double[] size = {1,1.5,2}; double Price = 0; switch (portion) { case 1: Price = this.unit_price * size[0]; break; case 2: Price = this.unit_price * size[1]; break; case 3: Price = this.unit_price * size[2]; break; default:; } return ((int)Math.round(Price)); }//计算菜品价格的方法,输入参数是点菜的份额(输入数据只能是1/2/3,代表小/中/大份) } //封装的读取器 public String getName() { return name; }//读取Name public int getUnit_price() { return unit_price; }//读取单价 public boolean getTT() { return TT; }// 读取特色菜的标志 public void setUnit_price(int unit_price) { this.unit_price = unit_price; } //重新定义单价 } //菜谱类:对应菜谱,包含饭店提供的所有菜的信息。 class Menu { Menu() { } ArrayList<Dish> dishes = new ArrayList<>(); //菜品数组,保存所有菜品信息 public Dish searchDish(String dishName) { boolean find = false; for (Dish dish : dishes){ if (dish.getName().equals(dishName)) { find = true; return dish; } //如果找到就返回dish的类信息 } System.out.println(dishName + " does not exist"); return null; //没有就返回null,用于判断是否找到 }//根据菜名在菜谱中查找菜品信息,返回Dish对象。 public void addDish(Dish dish) { //开始为返回Dish类???为什么??? if (search(dish.getName())) { Dish pp = searchDish(dish.getName()); pp.setUnit_price(dish.getUnit_price()); }//如果发现有该菜品则重新定义该单价 else { dishes.add(dish); }//没有发现则直接加入dishes列表里面去 } //添加一道菜品信息 public boolean search(String dishName) { boolean find = false; for (Dish dish : dishes){ if (dish.getName().equals(dishName)) { find = true; break; } //如果找到就返回boolean } return find; }//根据菜名在菜谱中查找菜品信息,返回Dish对象。 } //希望能够实现存储每一个table的基本信息 //包括该table的订单和记录 class Table { ArrayList<Integer> deletes = new ArrayList<>(); Menu menu = new Menu(); int num;//座序号 // int sum = 0;//总价钱 Calendar data = Calendar.getInstance();//时间 Order order;//订单 Record record;//记录 double discount = -1; int year,month,day,hours,minute,second; //时间存储Date的参数 int week; //用来存储星期数目的 boolean time_error = false; //保存time的格式对错 Table(String num , String Day , String Time,Menu menu) { this.num = Integer.parseInt(num); //保存序号 String[] dataString = Day.split("/"); //保存日期 this.year = Integer.parseInt(dataString[0]); this.month = Integer.parseInt(dataString[1]); this.day = Integer.parseInt(dataString[2]); String[] timeString = Time.split("/"); //保存时间 this.hours = Integer.parseInt(timeString[0]); this.minute = Integer.parseInt(timeString[1]); this.second = Integer.parseInt(timeString[2]); try { for (int i = 0 ; i < 3 ; i++) { if (timeString[i].length()>2) { time_error = true; break; } } if (year%1000==0) { System.out.println(num + " date error"); } else { Calendar data_start = Calendar.getInstance(); data_start.set(2022,1,1); Calendar data_end = Calendar.getInstance(); data_end.set(2023,12,31); data.set(year,month,day,hours,minute,second); if (data.after(data_end) || data.before(data_start)) { System.out.println("not a valid time period"); } else { boolean isFirstSunday = (data.getFirstDayOfWeek() == Calendar.SUNDAY); week = data.get(Calendar.DAY_OF_WEEK); //若一周第一天为星期天,则-1 if(isFirstSunday){ week = week - 1; if(week == 0){ week = 7; } } order = new Order(); //同时构造Order类和Record类 record = new Record(); this.menu = menu; } } } catch (Exception e1) { System.out.println(num + " date error"); } } public void GetToTalPrice() { if (discount<0) { //没有折扣,说明不在营业时间之内 System.out.println("table " + num + " out of opening hours"); } else { //有折扣,说明在营业时间之内 int primer_uuu = 0; int uuu = 0; for (int i = 0 ; i < order.records.size() ; i++) { if (!deletes.contains(order.records.get(i).orderNum)) { //把没有删除的进行相加 primer_uuu = primer_uuu + order.records.get(i).getPrice(); if (order.records.get(i).d.getTT()) { //是特色菜,特别处理折扣 uuu = uuu + (int)Math.round(primer_uuu * 0.7); } else { //不是特色菜 uuu = uuu + (int)Math.round(primer_uuu * discount); } } } // int sum_primer = (int)sum; // sum = (int)Math.round(sum*discount); System.out.println("table " + num + ": " + primer_uuu + " " + uuu); } } public void Discount() { if (week>=1 && week<=5) { //在工作日的时间内 if (hours>=17 && hours<20) { discount = 0.8; } else if (hours==20 && minute<30){ discount = 0.8; } else if (hours==20 && minute==30 && second==0){ discount = 0.8; } else if ((hours>=11 && hours <=13) || (hours==10 && minute>=30)){ discount = 0.6; } else if (hours==14 && minute<30) { discount = 0.6; } else if (hours==14 && minute==30 && second==0) { discount = 0.6; } } else { //周末 if (hours>=10 && hours<=20) { discount = 1.0; } else if (hours==9 && minute>=30){ discount = 1.0; } else if ((hours==21 && minute<30) || (hours==21 && minute==30 && second==0)){ discount = 1.0; } } } //判断是否在营业的时间内 } //订单类:保存用户点的所有菜的信息。 class Order { Order() {} //构造器 ArrayList<Record> records = new ArrayList<>();//保存订单上每一道的记录 public int getTotalPrice() { int sum = 0; for (Record pp : records){ sum=sum+pp.getPrice(); } return sum; //打算放到Table里面进行相对应的Date判断,以便进行相对应的日期优惠!!! //还是使用它,先计算总价位,再来对日期进行相对应的打折 }//计算订单的总价 public void addARecord(Record pp) { //原来为Record返回类型 records.add(pp); }//添加一条菜品信息到订单中。 public void delARecordByOrderNum(int orderNum) { //例子为int类??? if (findRecordByNum(orderNum)) { records.remove(find(orderNum)); }//找到删除 else { System.out.println("delete error"); }//未找到则输出对应的信息 }//根据序号删除一条记录 public boolean findRecordByNum(int orderNum) { //原来为void类型 for (Record pp : records) { if (pp.orderNum == orderNum) { return true; } } return false; }//根据序号查找一条记录 public Record find (int orderNum) { //可以通过订单进行记录的查找 for (Record pp : records) { if (pp.orderNum == orderNum) { return pp; } } return null; }//返回对应的Record类 public Record find_ByName(String name) { //通过名字查找Order中的Record for (Record pp : records) { if (pp.d.getName().equals(name)){ return pp; } } return null; } } //点菜记录类:保存订单上的一道菜品记录 class Record { Record(){ } //空构造器 Record(int orderNum,String dishName,int portion,int num) { //构造器 this.orderNum = orderNum; this.portion = portion; this.d = new Dish(dishName); this.num = num; setDish(dishName); } //非空构造器 int orderNum;//序号 Dish d;//菜品 public void setDish(String dishName) { d = new Dish(dishName); } int portion;//份额(1/2/3代表小/中/大份) int num; //份数,用来存储份数 public int getPrice() { int Price = 0; Price = d.getPrice(portion)*num; return Price; }//计价,计算本条记录的价格,返回int单个菜品的总价 //注意这里是未打折的总价 }
测试点如下

分析
1.必须得要对其中的输入格式进行准确分隔,如对菜单,table,点菜,带点菜等的输入进行分析,并且进行相对应的信息处理,对对应的类进行信息更新,同时对错误的输入信息进行区分和错误提醒。
2.这里的时间错误尤为的复杂,一是对工作日的判断,齐次是对优惠日的判断,最后是对开店时间的判断。在进行判断的同时还必须对不存在的日期进行对应的信息处理(如:2月30号和25:00分的错误情况)这里大概率是得要用try和catch来对错误的信息输入进行正确的错误判断。
3.必须对输入的信息进行判断,把对应的信息放入到对应的对象数据里,再在输出的时候进行提取。(就是因为这样子我这个题根本写不出来,太复杂了,或者是我太菜了)(后面的课程设计因为老师给我们讲了一下答案的大致框架,所有在写的时候就不会过于复杂)。
难点
1.得对数据处理的格式进行判断,这里我是使用双向关联的关系,使用的时候就极大的增加了我编写代码的负担。如果能换一个更好一点的框架说不定我就不会是24分了—_—...。
2.必须得对数据的类型进行准确的判断否则无法对输入的信息进行准确的分析和判断。
总结
1.如果在自己写的结构下无法顺利的展开自己的项目,那就尝试去看一下能不能改进一下自己的代码结构。
2.如果有不懂的最好去问一下牛逼的同学或者是老师,来帮助自己去解决这个问题,不要等着时间过去,就问不到了(就像我,我就是反面例子)。
7-1 菜单计价程序-5
比菜单计价程序-4多的部分
输入格式
代点菜信息包含:桌号 序号 菜品名称 口味度 份额 份数
信息处理
1、菜单输入时增加特色菜,特色菜的输入格式:菜品名+英文空格+口味类型+英文空格+基础价格+"T"
例如:麻婆豆腐 川菜 9 T
菜价的计算方法:
周一至周五 7折, 周末全价。
特色菜的口味类型:川菜、晋菜、浙菜
川菜增加辣度值:辣度0-5级;对应辣度水平为:不辣、微辣、稍辣、辣、很辣、爆辣;
晋菜增加酸度值,酸度0-4级;对应酸度水平为:不酸、微酸、稍酸、酸、很酸;
浙菜增加甜度值,甜度0-3级;对应酸度水平为:不甜、微甜、稍甜、甜;
例如:麻婆豆腐 川菜 9 T
输入订单记录时如果是特色菜,添加口味度(辣/酸/甜度)值,格式为:序号+英文空格+菜名+英文空格+口味度值+英文空格+份额+英文空格+份数
例如:1 麻婆豆腐 4 1 9
单条信息在处理时,如果口味度超过正常范围,输出"spicy/acidity/sweetness num out of range : "+口味度值,spicy/acidity/sweetness(辣度/酸度/甜度)根据菜品类型择一输出,例如:
acidity num out of range : 5
输出一桌的信息时,按辣、酸、甜度的顺序依次输出本桌菜各种口味的口味度水平,如果没有某个类型的菜,对应的口味(辣/酸/甜)度不输出,只输出已点的菜的口味度。口味度水平由口味度平均值确定,口味度平均值只综合对应口味菜系的菜计算,不做所有菜的平均。比如,某桌菜点了3份川菜,辣度分别是1、3、5;还有4份晋菜,酸度分别是,1、1、2、2,辣度平均值为3、酸度平均值四舍五入为2,甜度没有,不输出。
一桌信息的输出格式:table+英文空格+桌号+:+英文空格+当前桌的原始总价+英文空格+当前桌的计算折扣后总价+英文空格+"川菜"+数量+辣度+英文空格+"晋菜"+数量+酸度+英文空格+"浙菜"+数量+甜度。
如果整桌菜没有特色菜,则只输出table的基本信息,格式如下,注意最后加一个英文空格:
table+英文空格+桌号+:+英文空格+当前桌的原始总价+英文空格+当前桌的计算折扣后总价+英文空格
例如:table 1: 60 36 川菜 2 爆辣 浙菜 1 微甜
计算口味度时要累计本桌各类菜系所有记录的口味度总和(每条记录的口味度乘以菜的份数),再除以对应菜系菜的总份数,最后四舍五入。
注:本题要考虑代点菜的情况,当前桌点的菜要加上被其他桌代点的菜综合计算口味度平均值。
2、考虑客户订多桌菜的情况,输入时桌号时,增加用户的信息:
格式:table+英文空格+桌号+英文空格+":"+英文空格+客户姓名+英文空格+手机号+日期(格式:YYYY/MM/DD)+英文空格+ 时间(24小时制格式: HH/MM/SS)
例如:table 1 : tom 13670008181 2023/5/1 21/30/00
约束条件:客户姓名不超过10个字符,手机号11位,前三位必须是180、181、189、133、135、136其中之一。
输出结果时,先按要求输出每一桌的信息,最后按字母顺序依次输出每位客户需要支付的金额。不考虑各桌时间段的问题,同一个客户的所有table金额都要累加。
输出用户支付金额格式:
输入格式
菜名+口味类型+英文空格+基础价格
点菜记录格式:序号+英文空格+菜名+英文空格+辣/酸/甜度值+英文空格+份额+英文空格+份数 注:份额可输入(1/2/3), 1代表小份,2代表中份,3代表大份。辣/酸/甜度取值范围见题目中说明。
输出格式
格式:table+英文空格+桌号+“:”+英文空格+当前桌的计算折扣后总价+英文空格+辣度平均值+英文空格+酸度平均值+英文空格+甜度平均值+英文空格
最后按拼音顺序输出每位客户(不考虑客户同名或拼音相同的情况)的支付金额,格式: 用户姓名+英文空格+手机号+英文空格+支付总金额,按输入顺序排列。
难度
*******
SourceMonitor

代码如下
import java.util.Scanner; import java.util.ArrayList; import java.util.List; import java.util.Calendar; import java.util.Date; public class Main { public static void main(String[] args) { Work work = new Work(); work.play(); } } class Work { boolean Stop = false; //用来确定订单和菜单混杂在一起的情况 Menu menu; //单位菜单 Record record; //单位记录 Table table; //单位座位 ArrayList<Table> tables = new ArrayList<>(); //座位数组 Work() { //初始化,用来创建基本的类数组 //只能用一个菜单,别单位化 //对所有的table都开放 menu = new Menu(); //单位化,订单? Record应该是必须得单位化 record = new Record(); //定义一个存储Table的一个ArrayList —— tables的动态数组 tables = new ArrayList<>(); } public void play() { //play 菜单程序主要运行的程序 Scanner in = new Scanner(System.in); while (true) { String Line = in.nextLine(); if (Line.equals("end")) { break; } String[] LineArray = Line.split(" "); if (LineArray.length == 2) { //一个空格 if (LineArray[1].equals("delete")) { //删除订单 int num = Integer.parseInt(LineArray[0]); if (table.deletes.contains(num)) { System.out.println("deduplication " + num); continue; } else { Record pp =table.order.find(num); table.order.delARecordByOrderNum(num); table.deletes.add(num); } } else { //菜单添加 if (!Stop) { //菜单还没有输出完 try { int x = Integer.parseInt(LineArray[1]); } catch (Exception e1) { System.out.println("wrong format"); continue; } Dish dish = new Dish(LineArray[0],Integer.parseInt(LineArray[1])); menu.addDish(dish); } else { System.out.println("invalid dish"); } } //continue; } else if (LineArray.length == 3) { if (LineArray[2].equals("T")) { //特色菜 Dish dish = new Dish(LineArray[0],Integer.parseInt(LineArray[1]),LineArray[2]); menu.addDish(dish); } } else if (LineArray.length == 4) { //三个空格,四个数据 if (LineArray[0].equals("table")) { //座号 try { table = new Table(LineArray[1], LineArray[2], LineArray[3],menu); if (table.time_error) { int yy = Integer.parseInt(LineArray[2]); //用来故意报错 } table.Discount(); //设定该table的折扣 System.out.println("table " + table.num + ": "); tables.add(table); Stop = true; } catch (Exception e2) { System.out.println("wrong format"); break; } } else { //添加订单 int number = Integer.parseInt(LineArray[0]); int portion = Integer.parseInt(LineArray[2]); int num = Integer.parseInt(LineArray[3]); if ((3<portion && portion<10)|| portion<1) { System.out.println(number + " portion out of range " + portion); if (num>15) { System.out.println(number + " num out of range " + num); } continue; } if (portion>=10) { System.out.println("not a valid time period"); if (num>15) { System.out.println(number + " num out of range " + num); } continue; } if (num>15) { System.out.println(number + " num out of range " + num); continue; } record = new Record(number, LineArray[1], portion, num); //初始化单位record的序号,菜名,份额,数量 if (menu.searchDish(LineArray[1]) != null) { //有该菜 Record oop = table.order.find_ByName(LineArray[1]); //查看Order中是否有该菜名的Record if (oop!=null && oop.portion == portion) { //如果有该菜名的Record并且portion相等 oop.num += num; continue; } record.d = menu.searchDish(LineArray[1]); //获得该菜名的Dish类 table.order.addARecord(record); //答案在if的前面??? int under_number = table.order.records.size()-1; //存储最后一个的table的下标 //往record中添加菜品,提供单价 //在record里面添加菜品的单价和名字 table.order.records.get(under_number).d = menu.searchDish(LineArray[1]); //获取record的单个总价 int price = table.order.records.get(under_number).getPrice(); //输出单个record的表出 System.out.println(table.order.records.get(under_number).orderNum + " " + LineArray[1] + " " +price); } } //continue; } else if (LineArray.length == 5) { //代点菜 try { int number = Integer.parseInt(LineArray[1]); int portion = Integer.parseInt(LineArray[3]); int num = Integer.parseInt(LineArray[4]); record = new Record(number, LineArray[2], portion, num); if (menu.searchDish(LineArray[2])!=null) { //菜单内有该菜品 //添加record table.order.addARecord(record); int under_number = table.order.records.size()-1; //设定record的菜品单价 table.order.records.get(under_number).d = menu.searchDish(LineArray[2]); int price = table.order.records.get(under_number).getPrice(); System.out.println(number + " table " + table.num + " pay for table " + LineArray[0] + price); } } catch (Exception e2){ System.out.println("wrong format"); break; } } } for (int i = 0 ; i < tables.size() ; i++) { tables.get(i).GetToTalPrice(); } } } // 菜品类:对应菜谱上一道菜的信息。 class Dish { //封装 Dish() { } //空构造器 Dish(String dishName) { this.name = dishName; } Dish(String dishName , int unit_price) { this.name = dishName; this.unit_price = unit_price; } //普通菜的构造器 Dish(String dishName , int unit_price , String pp) { this.name = dishName; this.unit_price = unit_price; if (pp.equals("T")) { TT = true; } } //特色菜的构造器 private String name;//菜品名称 private int unit_price; //单价 private boolean TT = false; //特色菜标志 public int getPrice(int portion) { double[] size = {1,1.5,2}; double Price = 0; switch (portion) { case 1: Price = this.unit_price * size[0]; break; case 2: Price = this.unit_price * size[1]; break; case 3: Price = this.unit_price * size[2]; break; default:; } return ((int)Math.round(Price)); }//计算菜品价格的方法,输入参数是点菜的份额(输入数据只能是1/2/3,代表小/中/大份) } //封装的读取器 public String getName() { return name; }//读取Name public int getUnit_price() { return unit_price; }//读取单价 public boolean getTT() { return TT; }// 读取特色菜的标志 public void setUnit_price(int unit_price) { this.unit_price = unit_price; } //重新定义单价 } //菜谱类:对应菜谱,包含饭店提供的所有菜的信息。 class Menu { Menu() { } ArrayList<Dish> dishes = new ArrayList<>(); //菜品数组,保存所有菜品信息 public Dish searchDish(String dishName) { boolean find = false; for (Dish dish : dishes){ if (dish.getName().equals(dishName)) { find = true; return dish; } //如果找到就返回dish的类信息 } System.out.println(dishName + " does not exist"); return null; //没有就返回null,用于判断是否找到 }//根据菜名在菜谱中查找菜品信息,返回Dish对象。 public void addDish(Dish dish) { //开始为返回Dish类???为什么??? if (search(dish.getName())) { Dish pp = searchDish(dish.getName()); pp.setUnit_price(dish.getUnit_price()); }//如果发现有该菜品则重新定义该单价 else { dishes.add(dish); }//没有发现则直接加入dishes列表里面去 } //添加一道菜品信息 public boolean search(String dishName) { boolean find = false; for (Dish dish : dishes){ if (dish.getName().equals(dishName)) { find = true; break; } //如果找到就返回boolean } return find; }//根据菜名在菜谱中查找菜品信息,返回Dish对象。 } //希望能够实现存储每一个table的基本信息 //包括该table的订单和记录 class Table { ArrayList<Integer> deletes = new ArrayList<>(); Menu menu = new Menu(); int num;//座序号 // int sum = 0;//总价钱 Calendar data = Calendar.getInstance();//时间 Order order;//订单 Record record;//记录 double discount = -1; int year,month,day,hours,minute,second; //时间存储Date的参数 int week; //用来存储星期数目的 boolean time_error = false; //保存time的格式对错 Table(String num , String Day , String Time,Menu menu) { this.num = Integer.parseInt(num); //保存序号 String[] dataString = Day.split("/"); //保存日期 this.year = Integer.parseInt(dataString[0]); this.month = Integer.parseInt(dataString[1]); this.day = Integer.parseInt(dataString[2]); String[] timeString = Time.split("/"); //保存时间 this.hours = Integer.parseInt(timeString[0]); this.minute = Integer.parseInt(timeString[1]); this.second = Integer.parseInt(timeString[2]); try { for (int i = 0 ; i < 3 ; i++) { if (timeString[i].length()>2) { time_error = true; break; } } if (year%1000==0) { System.out.println(num + " date error"); } else { Calendar data_start = Calendar.getInstance(); data_start.set(2022,1,1); Calendar data_end = Calendar.getInstance(); data_end.set(2023,12,31); data.set(year,month,day,hours,minute,second); if (data.after(data_end) || data.before(data_start)) { System.out.println("not a valid time period"); } else { boolean isFirstSunday = (data.getFirstDayOfWeek() == Calendar.SUNDAY); week = data.get(Calendar.DAY_OF_WEEK); //若一周第一天为星期天,则-1 if(isFirstSunday){ week = week - 1; if(week == 0){ week = 7; } } order = new Order(); //同时构造Order类和Record类 record = new Record(); this.menu = menu; } } } catch (Exception e1) { System.out.println(num + " date error"); } } public void GetToTalPrice() { if (discount<0) { //没有折扣,说明不在营业时间之内 System.out.println("table " + num + " out of opening hours"); } else { //有折扣,说明在营业时间之内 int primer_uuu = 0; int uuu = 0; for (int i = 0 ; i < order.records.size() ; i++) { if (!deletes.contains(order.records.get(i).orderNum)) { //把没有删除的进行相加 primer_uuu = primer_uuu + order.records.get(i).getPrice(); if (order.records.get(i).d.getTT()) { //是特色菜,特别处理折扣 uuu = uuu + (int)Math.round(primer_uuu * 0.7); } else { //不是特色菜 uuu = uuu + (int)Math.round(primer_uuu * discount); } } } // int sum_primer = (int)sum; // sum = (int)Math.round(sum*discount); System.out.println("table " + num + ": " + primer_uuu + " " + uuu); } } public void Discount() { if (week>=1 && week<=5) { //在工作日的时间内 if (hours>=17 && hours<20) { discount = 0.8; } else if (hours==20 && minute<30){ discount = 0.8; } else if (hours==20 && minute==30 && second==0){ discount = 0.8; } else if ((hours>=11 && hours <=13) || (hours==10 && minute>=30)){ discount = 0.6; } else if (hours==14 && minute<30) { discount = 0.6; } else if (hours==14 && minute==30 && second==0) { discount = 0.6; } } else { //周末 if (hours>=10 && hours<=20) { discount = 1.0; } else if (hours==9 && minute>=30){ discount = 1.0; } else if ((hours==21 && minute<30) || (hours==21 && minute==30 && second==0)){ discount = 1.0; } } } //判断是否在营业的时间内 } //订单类:保存用户点的所有菜的信息。 class Order { Order() {} //构造器 ArrayList<Record> records = new ArrayList<>();//保存订单上每一道的记录 public int getTotalPrice() { int sum = 0; for (Record pp : records){ sum=sum+pp.getPrice(); } return sum; //打算放到Table里面进行相对应的Date判断,以便进行相对应的日期优惠!!! //还是使用它,先计算总价位,再来对日期进行相对应的打折 }//计算订单的总价 public void addARecord(Record pp) { //原来为Record返回类型 records.add(pp); }//添加一条菜品信息到订单中。 public void delARecordByOrderNum(int orderNum) { //例子为int类??? if (findRecordByNum(orderNum)) { records.remove(find(orderNum)); }//找到删除 else { System.out.println("delete error"); }//未找到则输出对应的信息 }//根据序号删除一条记录 public boolean findRecordByNum(int orderNum) { //原来为void类型 for (Record pp : records) { if (pp.orderNum == orderNum) { return true; } } return false; }//根据序号查找一条记录 public Record find (int orderNum) { //可以通过订单进行记录的查找 for (Record pp : records) { if (pp.orderNum == orderNum) { return pp; } } return null; }//返回对应的Record类 public Record find_ByName(String name) { //通过名字查找Order中的Record for (Record pp : records) { if (pp.d.getName().equals(name)){ return pp; } } return null; } } //点菜记录类:保存订单上的一道菜品记录 class Record { Record(){ } //空构造器 Record(int orderNum,String dishName,int portion,int num) { //构造器 this.orderNum = orderNum; this.portion = portion; this.d = new Dish(dishName); this.num = num; setDish(dishName); } //非空构造器 int orderNum;//序号 Dish d;//菜品 public void setDish(String dishName) { d = new Dish(dishName); } int portion;//份额(1/2/3代表小/中/大份) int num; //份数,用来存储份数 public int getPrice() { int Price = 0; Price = d.getPrice(portion)*num; return Price; }//计价,计算本条记录的价格,返回int单个菜品的总价 //注意这里是未打折的总价 }
测试点如下

分析
1.这里相比上面的菜单计价程序-4多了特色菜,多了名字和身份证,多了table的错误判断的因素。需要对对应的输出进行对应的数据处理。
2.相较于上次多了菜品和菜品的甜度,酸度,辣度,也加大的代码对信息的读入和处理量。
3.输入掺杂了上次的输入格式和这一次新的输入格式,需要在两种相似的输入格式进行细致的区分。
难点
1.输入的信息格式更多变,需要对输入的信息进行更细致的区分来进行相对应的信息处理。
2.对菜品也得要有更多的分类的信息的读取和读入。
总结
1.要留给难的题目足够的时间,去实现对应的功能,不要拖着,否则等到后面不再会有机会去自主完成这个作业。(这里我隆重的向我自己道歉,因为没有留给它太多的时间,那段时间不怎么学习编程,都把心思花在别的地方上去了,主要是手机短视频占了我太多的时间了,齐次我对自己的编程能力过分的自信,那时候以为自己做的出来,其实自己根本连一个框架都搭建的不大好,更何况是在细节的方面上,更是顽石一块。我的问题。(忏悔录))。
2.在开始着手去编写代码的时候,必须得先要考虑一下编写代码的大体结构,以便更好地,更流畅地去编写相对应的功能。如果前面的结构太差的话,后面的编写手感和进度都会成指线型上升,照成一步错步步错,还是得先在0-1这一个线段下好扎实的功夫,多去学习他人的代码结构和他人的代码风格,取其精华,去其糟粕来加强自己的代码编写的能力和代码的质量。
3.通过这两次的点菜程序作业,一个是0分,一个是24分,我在后面的时间,必须得要花跟多的时间去改善和完善我自己的代码习惯和代码质量。

浙公网安备 33010602011771号