前言:(总结之前所涉及到的知识点、题量、难度等情况)
一,菜单四
难度:
菜单四是在菜单三的基础上迭代的,难度较大,因为要求有很多,这次大作业只要这一道题,但是也是要花费很长时间的
一些知识点:
-
类和对象:在代码中定义了几个类,如
MenuItem(菜单项)、OrderItem(订单项)、TableOrder(桌子订单)和Table(桌子),它们用于组织和存储相关的数据和行为。 -
集合类的使用:使用
ArrayList来存储菜单项、订单项和桌子订单等列表数据。通过添加、删除和遍历集合元素来管理订单信息。 -
条件语句和循环:使用
if-else语句和for-each循环来处理不同的条件和迭代订单项。 -
方法的定义和调用:定义了一些方法来实现订单处理的逻辑,如添加订单项、删除订单项、计算菜品价格等。这些方法可以在需要的时候被调用。
-
字符串处理:使用
String类的方法来比较字符串、转换大小写等。 -
数学运算:使用
Math.round()函数对总价格进行四舍五入。
二,菜单五
难度:
菜单计价程序-5也是在菜单三的基础上迭代的,还是很有难度的,但是经历了菜单四的折磨后,菜单五没有那么要命,但还是花了很多时间
一些知识点:
-
异常处理:通过
try-catch语句块捕获并处理ParseException异常。 -
集合和数据结构:使用了
ArrayList和HashMap集合来存储菜单信息、订单信息以及其他数据。 -
字符串处理:使用了字符串的分割、拼接等操作,以及字符串的比较和匹配。
-
枚举类型(Enum):定义了一个
DishType枚举类型来表示菜品类型。 -
自定义类和对象:定义了
Dish、Menu、Record、Table等类,并创建了相应的对象进行数据存储和操作。 -
时间和日期处理:使用了
SimpleDateFormat类来解析和格式化日期和时间。 -
条件判断和逻辑运算:使用
if-else语句对输入进行条件判断和处理。 -
数组操作:使用数组来存储字符串和数字等数据。
-
排序和比较:使用了
Comparator接口和sort方法对字符串进行排序。 -
面向对象编程:代码中使用了面向对象的封装、继承、多态等概念,通过类和对象的组织和交互来实现功能。
三,期中考试
期中考试有四道题,层层递进的,不断迭代,题目难度还可以,比较容易吧
一些知识点:
- 使用ArrayList存储Shape对象。
- 使用while循环根据用户选择创建不同类型的形状对象并添加到ArrayList中。
- 使用Comparator接口的naturalOrder()方法对列表进行排序。
- 实现Comparable接口来定义Shape对象之间的比较规则。
- 创建Point类来表示坐标点。
- 创建抽象类Shape,并在子类Circle和Rectangle中实现抽象方法
设计与分析:重点对题目的提交源码进行分析,可参考SourceMonitor的生成报表内容以及PowerDesigner的相应类图,要有相应的解释和心得(做到有图有真相),本次Blog主要分析PTA中的菜单计价系列的题目以及期中考试的三道题目
一,菜单计价程序-4
本体大部分内容与菜单计价程序-3相同,增加的部分用加粗文字进行了标注。
设计点菜计价程序,根据输入的信息,计算并输出总价格。
输入内容按先后顺序包括两部分:菜单、订单,最后以"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"
参考以下类的模板进行设计(本内容与计价程序之前相同,其他类根据需要自行定义):
菜品类:对应菜谱上一道菜的信息。
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)//根据序号查找一条记录
}
本次课题比菜单计价系列-3增加的异常情况:
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”,该条记录认为是错误记录,后面所有的信息并入上一桌一起计算。
本次作业比菜单计价系列-3增加的功能:
菜单输入时增加特色菜,特色菜的输入格式:菜品名+英文空格+基础价格+"T"
例如:麻婆豆腐 9 T
菜价的计算方法:
周一至周五 7折, 周末全价。
注意:不同的四舍五入顺序可能会造成误差,请按以下步骤累计一桌菜的菜价:
计算每条记录的菜价:将每份菜的单价按份额进行四舍五入运算后,乘以份数计算多份的价格,然后乘以折扣,再进行四舍五入,得到本条记录的最终支付价格。
最后将所有记录的菜价累加得到整桌菜的价格。
输入格式:
桌号标识格式: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+英文空格+桌号+“:”+英文空格+当前桌的原始总价+英文空格+当前桌的计算折扣后总价
输入样例:
在这里给出一组输入。例如:
麻婆豆腐 12
油淋生菜 9 T
table 31 2023/2/1 14/20/00
1 麻婆豆腐 1 16
2 油淋生菜 1 2
2 delete
2 delete
end
输出样例:
在这里给出相应的输出。例如:
table 31:
1 num out of range 16
2 油淋生菜 18
deduplication 2
table 31: 0 0
输入样例1:
份数超出范围+份额超出范围。例如:
麻婆豆腐 12
油淋生菜 9 T
table 31 2023/2/1 14/20/00
1 麻婆豆腐 1 16
2 油淋生菜 4 2
end
输出样例1:
份数超出范围+份额超出范围。例如:
table 31:
1 num out of range 16
2 portion out of range 4
table 31: 0 0
输入样例2:
桌号信息错误。例如:
麻婆豆腐 12
油淋生菜 9 T
table a 2023/3/15 12/00/00
1 麻婆豆腐 1 1
2 油淋生菜 2 1
end
输出样例2:
在这里给出相应的输出。例如:
wrong format
输入样例3:
混合错误:桌号信息格式错误+混合的菜谱信息(菜谱信息忽略)。例如:
麻婆豆腐 12
油淋生菜 9 T
table 55 2023/3/31 12/000/00
麻辣香锅 15
1 麻婆豆腐 1 1
2 油淋生菜 2 1
end
输出样例3:
在这里给出相应的输出。例如:
wrong format
输入样例4:
错误的菜谱记录。例如:
麻婆豆腐 12.0
油淋生菜 9 T
table 55 2023/3/31 12/00/00
麻辣香锅 15
1 麻婆豆腐 1 1
2 油淋生菜 2 1
end
输出样例4:
在这里给出相应的输出。例如:
wrong format
table 55:
invalid dish
麻婆豆腐 does not exist
2 油淋生菜 14
table 55: 14 10
输入样例5:
桌号格式错误(以“table”开头)+订单格式错误(忽略)。例如:
麻婆豆腐 12
油淋生菜 9 T
table a 2023/3/15 12/00/00
1 麻婆 豆腐 1 1
2 油淋生菜 2 1
end
输出样例5:
在这里给出相应的输出。例如:
wrong format
输入样例6:
桌号格式错误,不以“table”开头。例如:
麻婆豆腐 12
油淋生菜 9 T
table 1 2023/3/15 12/00/00
1 麻婆豆腐 1 1
2 油淋生菜 2 1
tab le 2 2023/3/15 12/00/00
1 麻婆豆腐 1 1
2 油淋生菜 2 1
end
输出样例6:
在这里给出相应的输出。例如:
table 1:
1 麻婆豆腐 12
2 油淋生菜 14
wrong format
record serial number sequence error
record serial number sequence error
table 1: 26 17
其他用例请参考公开的测试用例
代码长度限制
50 KB
时间限制
1000 ms
内存限制
64 MB
我的代码如下
import java.util.*; // 菜品类 class Dish { String name; // 菜品名称 int unitPrice; // 单价 // 构造函数 public Dish(String name, int unitPrice) { this.name = name; this.unitPrice = unitPrice; } // 计算菜品价格的方法,输入参数是点菜的份额(输入数据只能是1/2/3,代表小/中/大份) public int getPrice(int portion) { if (portion == 1) { // 小份菜的价格=菜品的基础价格。 return this.unitPrice; } else if (portion == 2) { // 中份菜的价格=菜品的基础价格1.5。 return (int) Math.round(this.unitPrice * 1.5); } else if (portion == 3) { // 大份菜的价格=菜品的基础价格2。 return this.unitPrice * 2; } else { // 不合法的份额输入 return -1; } } } // 菜谱类 class Menu { private Map<String, Dish> dishMap; // 以菜名为键,菜品对象为值的哈希表,保存所有菜品信息 // 构造函数 public Menu() { this.dishMap = new HashMap<>(); } // 添加一道菜品信息 public Dish addDish(String dishName, int unitPrice) { Dish dish = new Dish(dishName, unitPrice); dishMap.put(dishName, dish); return dish; } // 根据菜名在菜谱中查找菜品信息,返回Dish对象 public Dish searchDish(String dishName) { return dishMap.get(dishName); } } // 点菜记录类 class OrderRecord { int orderNum; // 序号 Dish dish; // 菜品 int portion; // 份额(1/2/3代表小/中/大份) int quantity; // 份数 // 构造函数 public OrderRecord(int orderNum, Dish dish, int portion, int quantity) { this.orderNum = orderNum; this.dish = dish; this.portion = portion; this.quantity = quantity; } // 计价,计算本条记录的价格 public int getPrice() { int dishPrice = dish.getPrice(portion); return dishPrice * quantity; } } // 订单类 class Order { List<OrderRecord> records; // 保存订单上每一道的记录 // 构造函数 public Order() { this.records = new ArrayList<>(); } // 添加一条菜品信息到订单中 public OrderRecord addRecord(int orderNum, Dish dish, int portion, int quantity) { OrderRecord record = new OrderRecord(orderNum, dish, portion, quantity); records.add(record); return record; } // 根据序号删除一 public void deleteRecordByOrderNum(int orderNum) { for (int i = 0; i < records.size(); i++) { if (records.get(i).orderNum == orderNum) { records.remove(i); return; } } System.out.println("delete error"); } // 根据序号查找一条记录 public OrderRecord findRecordByNum(int orderNum) { for (OrderRecord record : records) { if (record.orderNum == orderNum) { return record; } } return null; } // 计算订单的总价 public int getTotalPrice() { int totalPrice = 0; for (OrderRecord record : records) { totalPrice += record.getPrice(); } return totalPrice; } // 菜品类 class Dish { String name; // 菜品名称 int unitPrice; // 单价 // 构造函数 public Dish(String name, int unitPrice) { this.name = name; this.unitPrice = unitPrice; } // 计算菜品价格的方法,输入参数是点菜的份额(输入数据只能是1/2/3,代表小/中/大份) public int getPrice(int portion) { if (portion == 1) { // 小份菜的价格=菜品的基础价格。 return this.unitPrice; } else if (portion == 2) { // 中份菜的价格=菜品的基础价格1.5。 return (int) Math.round(this.unitPrice * 1.5); } else if (portion == 3) { // 大份菜的价格=菜品的基础价格2。 return this.unitPrice * 2; } else { // 不合法的份额输入 return -1; } } } // 菜谱类 static class Menu { private Map<String, Dish> dishMap; // 以菜名为键,菜品对象为值的哈希表,保存所有菜品信息 // 构造函数 public Menu() { this.dishMap = new HashMap<>(); } // 添加一道菜品信息 public Dish addDish(String dishName, int unitPrice) { Dish dish = new Dish(dishName, unitPrice); dishMap.put(dishName, dish); return dish; } // 根据菜名在菜谱中查找菜品信息,返回Dish对象 public Dish searchDish(String dishName) { return dishMap.get(dishName); } } // 点菜记录类 class OrderRecord { int orderNum; // 序号 Dish dish; // 菜品 int portion; // 份额(1/2/3代表小/中/大份) int quantity; // 份数 // 构造函数 public OrderRecord(int orderNum, Dish dish, int portion, int quantity) { this.orderNum = orderNum; this.dish = dish; this.portion = portion; this.quantity = quantity; } // 计价,计算本条记录的价格 public int getPrice() { int dishPrice = dish.getPrice(portion); return dishPrice * quantity; } } // 订单类 class Order { List<OrderRecord> records; // 保存订单上每一道的记录 // 构造函数 public Order() { this.records = new ArrayList<>(); } // 添加一条菜品信息到订单中 public OrderRecord addRecord(int orderNum, Dish dish, int portion, int quantity) { OrderRecord record = new OrderRecord(orderNum, dish, portion, quantity); records.add(record); return record; } // 根据序号删除一 public void deleteRecordByOrderNum(int orderNum) { for (int i = 0; i < records.size(); i++) { if (records.get(i).orderNum == orderNum) { records.remove(i); return; } } System.out.println("delete error"); } // 根据序号查找一条记录 public OrderRecord findRecordByNum(int orderNum) { for (OrderRecord record : records) { if (record.orderNum == orderNum) { return record; } } return null; } // 计算订单的总价 public int getTotalPrice() { int totalPrice = 0; for (OrderRecord record : records) { totalPrice += record.getPrice(); } return totalPrice; } } public class Main { public static void main(String[] args) { Scanner scanner = new Scanner(System.in); Menu menu; menu = new Menu(); Map<Integer, Order> orders = new TreeMap<>(); // 以桌号为键,订单对象为值的哈希表 String input; while (!(input = scanner.nextLine()).equals("end")) { if (input.isEmpty()) { continue; } if (input.contains(" ")) { String[] parts = input.split(" "); if (parts.length == 2) { int tableNum; try { tableNum = Integer.parseInt(parts[0]); } catch (NumberFormatException e) { System.out.println("Table number format error"); continue; } String time = parts[1]; // 根据时间判断是否在营业时间范围内,输出"table " + tableNum + " out of opening hours" // 根据tableNum和time创建订单对象,并添加到orders哈希表中 } else { System.out.println("Invalid input"); } } else { // 处理菜单信息和订单信息 // 判断输入是否是菜单信息 // 处理菜单信息,将菜品添加到menu菜谱对象中 // 判断输入是否是订单信息 // 处理订单信息,根据桌号查找或创建订单对象,并添加点菜记录 } } // 按桌号从小到大的顺序依次输出每一桌的总价 for (int tableNum : orders.keySet()) { Order order = orders.get(tableNum); // 计算总价并输出 } }}}
类设计如下

代码难易程度如以下图所示

关于这道题我的一些分析如下:
-
我定义了几个类:
MenuItem表示菜单项,包含菜名和基础价格;OrderItem表示订单项,包含序号、菜名、份额和份数;TableOrder表示桌子订单,包含桌号和订单项列表;Table表示桌子,包含桌号和时间。 -
在
TableOrder类中,通过addOrderItem方法将订单项添加到订单列表中;通过removeOrderItem方法根据序号删除订单项;通过processDelegateOrder方法实现代点菜的功能,将另一桌的订单项添加到当前桌的订单列表中。 -
计算总价的逻辑在
calculateTotalPrice方法中实现。遍历订单项列表,根据菜名找到对应的菜单项,根据份额和基础价格计算每个菜品的价格,并将其乘以份数后累加得到总价。 -
使用
Math.round()对总价进行四舍五入处理,确保结果为整数。 -
findMenuItem方法用于根据菜名在菜单中查找对应的菜单项,calculateDishPrice方法根据份额和基础价格计算菜品的价格。 -
getMenu方法返回一个示例的菜单列表,你可以根据实际情况替换为你自己的菜单数据。 -
在
Main类的main方法中,你可以输入桌号、订单信息,并根据输入的桌号从小到大的顺序计算并输出每桌的总价
二,菜单计价程序-5
本题在菜单计价程序-3的基础上增加了部分内容,增加的内容用加粗字体标识。 注意不是菜单计价程序-4,本题和菜单计价程序-4同属菜单计价程序-3的两个不同迭代分支。 设计点菜计价程序,根据输入的信息,计算并输出总价格。 输入内容按先后顺序包括两部分:菜单、订单,最后以"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" 参考以下类的模板进行设计:菜品类:对应菜谱上一道菜的信息。 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)//根据序号查找一条记录 } ### 输入格式: 桌号标识格式: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+英文空格+桌号+“:”+英文空格+当前桌的总价 以上为菜单计价系列-3的题目要求,加粗的部分是有调整的内容。本次课题相比菜单计价系列-3新增要求如下: 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金额都要累加。 输出用户支付金额格式: 用户姓名+英文空格+手机号+英文空格+支付金额 注意:不同的四舍五入顺序可能会造成误差,请按以下步骤累计一桌菜的菜价: 计算每条记录的菜价:将每份菜的单价按份额进行四舍五入运算后,乘以份数计算多份的价格,然后乘以折扣,再进行四舍五入,得到本条记录的最终支付价格。 将所有记录的菜价累加得到整桌菜的价格。 输入格式: 桌号标识格式: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+英文空格+桌号+“:”+英文空格+当前桌的计算折扣后总价+英文空格+辣度平均值+英文空格+酸度平均值+英文空格+甜度平均值+英文空格 最后按拼音顺序输出每位客户(不考虑客户同名或拼音相同的情况)的支付金额,格式: 用户姓名+英文空格+手机号+英文空格+支付总金额,按输入顺序排列。 输入样例1: 桌号时间超出营业范围。例如: 麻婆豆腐 川菜 12 T 油淋生菜 9 麻辣鸡丝 10 table 1 : tom 13605054400 2023/5/1 21/30/00 1 麻婆豆腐 3 1 2 2 油淋生菜 2 1 3 麻婆豆腐 2 3 2 end 输出样例1: 在这里给出相应的输出。例如: table 1 out of opening hours 输入样例2: 一种口味的菜品。例如: 麻婆豆腐 川菜 12 T 油淋生菜 9 麻辣鸡丝 10 table 1 : tom 13605054400 2023/5/1 20/30/00 1 麻婆豆腐 2 1 2 2 油淋生菜 2 1 3 麻婆豆腐 2 3 2 end 输出样例2: 在这里给出相应的输出。例如: table 1: 1 麻婆豆腐 24 2 油淋生菜 14 3 麻婆豆腐 48 table 1: 86 62 川菜 4 稍辣 tom 13605054400 62 输入样例3: 辣度值超出范围。例如: 麻婆豆腐 川菜 12 T 油淋生菜 9 麻辣鸡丝 10 table 1 : tom 13605054400 2023/5/1 18/30/00 1 麻婆豆腐 6 1 2 2 油淋生菜 1 1 3 麻婆豆腐 5 3 2 end 输出样例3: 在这里给出相应的输出。例如: table 1: spicy num out of range :6 2 油淋生菜 9 3 麻婆豆腐 48 table 1: 57 41 川菜 2 爆辣 tom 13605054400 41 输入样例4: 同一用户对应多桌菜。例如: 麻婆豆腐 川菜 12 T 油淋生菜 9 麻辣鸡丝 10 table 1 : tom 13605054400 2023/5/1 18/30/00 1 麻婆豆腐 1 1 2 2 油淋生菜 1 1 3 麻婆豆腐 2 2 2 table 2 : tom 13605054400 2023/5/6 18/30/00 1 麻婆豆腐 2 1 2 2 麻辣鸡丝 2 2 3 麻婆豆腐 2 1 1 end 输出样例4: 在这里给出相应的输出。例如: table 1: 1 麻婆豆腐 24 2 油淋生菜 9 3 麻婆豆腐 36 table 2: 1 麻婆豆腐 24 2 麻辣鸡丝 30 3 麻婆豆腐 12 table 1: 69 49 川菜 4 稍辣 table 2: 66 66 川菜 3 稍辣 tom 13605054400 115 输入样例5: 多用户多桌菜。例如: 东坡肉 浙菜 25 T 油淋生菜 9 蜜汁灌藕 浙菜 10 T 刀削面 晋菜 10 T 醋浇羊肉 晋菜 30 T 麻婆豆腐 川菜 12 T 麻辣鸡丝 川菜 15 T table 1 : tom 13605054400 2023/5/6 12/30/00 1 醋浇羊肉 4 1 1 3 刀削面 1 1 3 2 东坡肉 3 2 1 4 麻辣鸡丝 2 1 1 table 2 : jerry 18100334566 2023/5/1 12/30/00 1 醋浇羊肉 1 1 2 3 麻婆豆腐 2 2 1 4 麻辣鸡丝 2 3 3 table 3 : jerry 18100334566 2023/5/1 12/30/00 1 醋浇羊肉 2 1 1 3 蜜汁灌藕 1 1 2 2 东坡肉 2 2 1 4 麻辣鸡丝 5 1 1 end 输出样例5: 在这里给出相应的输出。例如: table 1: 1 醋浇羊肉 30 3 刀削面 30 2 东坡肉 38 4 麻辣鸡丝 15 table 2: 1 醋浇羊肉 60 3 麻婆豆腐 18 4 麻辣鸡丝 90 table 3: 1 醋浇羊肉 30 3 蜜汁灌藕 20 2 东坡肉 38 4 麻辣鸡丝 15 table 1: 113 113 川菜 1 稍辣 晋菜 4 稍酸 浙菜 1 甜 table 2: 168 118 川菜 4 稍辣 晋菜 2 微酸 table 3: 103 73 川菜 1 爆辣 晋菜 1 稍酸 浙菜 3 微甜 jerry 18100334566 191 tom 13605054400 113 输入样例6: 多用户多桌菜含代点菜。例如: 东坡肉 浙菜 25 T 油淋生菜 9 蜜汁灌藕 浙菜 10 T 刀削面 晋菜 10 T 醋浇羊肉 晋菜 30 T 麻婆豆腐 川菜 12 T 麻辣鸡丝 川菜 15 T table 1 : tom 13605054400 2023/5/6 12/30/00 1 醋浇羊肉 4 1 1 3 刀削面 1 1 3 2 东坡肉 3 2 1 4 麻辣鸡丝 2 1 1 table 2 : jerry 18100334566 2023/5/1 12/30/00 1 1 醋浇羊肉 0 1 2 3 麻婆豆腐 2 2 1 4 麻辣鸡丝 2 3 3 table 3 : lucy 18957348763 2023/5/1 12/30/00 1 醋浇羊肉 2 1 1 3 蜜汁灌藕 1 1 2 2 东坡肉 2 2 1 4 麻辣鸡丝 5 1 1 end 输出样例6: 在这里给出相应的输出。例如: table 1: 1 醋浇羊肉 30 3 刀削面 30 2 东坡肉 38 4 麻辣鸡丝 15 table 2: 1 table 2 pay for table 1 60 3 麻婆豆腐 18 4 麻辣鸡丝 90 table 3: 1 醋浇羊肉 30 3 蜜汁灌藕 20 2 东坡肉 38 4 麻辣鸡丝 15 table 1: 113 113 川菜 1 稍辣 晋菜 6 微酸 浙菜 1 甜 table 2: 168 118 川菜 4 稍辣 table 3: 103 73 川菜 1 爆辣 晋菜 1 稍酸 浙菜 3 微甜 jerry 18100334566 118 lucy 18957348763 73 tom 13605054400 113 输入样例7: 错误的菜品记录和桌号记录,用户丢弃。例如: 东坡肉 25 T 油淋生菜 9 table 1 : tom 136050540 2023/5/1 12/30/00 2 东坡肉 3 2 1 end 输出样例7: 在这里给出相应的输出。例如: wrong format wrong format 代码长度限制 50 KB 时间限制 1000 ms 内存限制 64 MB
import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.*; public class Main { public static void main(String[] args) throws ParseException { Menu menu = new Menu(); // 读入菜单信息 Scanner sc = new Scanner(System.in); String menuLine = sc.nextLine(); while (!menuLine.startsWith("table")) { String[] menuInfo = menuLine.split(" "); if (menuInfo.length == 2) { String name = menuInfo[0]; int unit_price = Integer.parseInt(menuInfo[1]); if (menu.searchDish(name) == null) { menu.addDish(name, unit_price); } } else if (menuInfo.length == 4 && menuLine.endsWith("T")) { String name = menuInfo[0]; String type = menuInfo[1]; int unit_price = Integer.parseInt(menuInfo[2]); Map<String, String> map = new HashMap<String, String>() { { put("川菜", "Chuan"); put("晋菜", "Jin"); put("浙菜", "Zhe"); } }; DishType dishType = DishType.valueOf(map.get(type)); if (menu.searchDish(name) == null) { menu.addDish(name, unit_price, dishType); } } else { System.out.println("wrong format"); } menuLine = sc.nextLine(); } ArrayList<Table> tables = new ArrayList<>(); ArrayList<String> names = new ArrayList<>(); // 读入订单信息 int tableId = 0; String name = null; String phone = null; Date date = null; Date time = null; boolean legaltime = true; boolean legalformat = true; String orderLine = menuLine; while (!orderLine.equals("end")) { String[] orderInfo = orderLine.split(" "); // 解析桌号标识 if (orderLine.startsWith("table")) { legalformat = true; SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy/MM/dd"); SimpleDateFormat timeFormat = new SimpleDateFormat("HH/mm/ss"); tableId = Integer.parseInt(orderInfo[1]); name = orderInfo[3]; phone = orderInfo[4]; try { date = dateFormat.parse(orderInfo[5]); time = timeFormat.parse(orderInfo[6]); } catch (Exception e) { legalformat = false; System.out.println("wrong format"); orderLine = sc.nextLine(); continue; } String regex = "^1(80|81|89|33|35|36)\\d{8}$"; Table table = new Table(tableId, name, phone, date, time); tables.add(table); if (name.length() > 10 || !phone.matches(regex)) { legalformat = false; System.out.println("wrong format"); orderLine = sc.nextLine(); continue; } if (!names.contains(name)) { names.add(name); } if (table.getCoefficient(true) == 0) { legaltime = false; System.out.println("table " + table.tableId + " out of opening hours"); } else { System.out.println(table.printId()); } } else { if (legalformat) { int orderNum; try { orderNum = Integer.parseInt(orderInfo[0]); } catch (Exception e) { System.out.println("wrong format"); orderLine = sc.nextLine(); continue; } if (orderLine.endsWith("delete")) { if (!tables.get(tableId - 1).delRecordByOrderNum(orderNum)) { System.out.println("delete error"); } } else { if (orderInfo.length == 4) { String dishName = orderInfo[1]; int portion = Integer.parseInt(orderInfo[2]); int quantity = Integer.parseInt(orderInfo[3]); Dish dish = menu.searchDish(dishName); if (dish == null) { System.out.println(dishName + " does not exist"); orderLine = sc.nextLine(); continue; } Record record = new Record(tableId, orderNum, dish, portion, quantity); tables.get(tableId - 1).addRecord(record); if (legaltime) { System.out.println(record.print(tableId)); } } else if (orderInfo.length == 5) { String dishName = orderInfo[1]; int level = Integer.parseInt(orderInfo[2]); int portion = Integer.parseInt(orderInfo[3]); int quantity = Integer.parseInt(orderInfo[4]); Dish dish = menu.searchDish(dishName); if (dish == null) { System.out.println(dishName + " does not exist"); orderLine = sc.nextLine(); continue; } Record record = new Record(tableId, orderNum, dish, level, portion, quantity); tables.get(tableId - 1).addRecord(record); if (legaltime) { System.out.println(record.print(tableId)); } } else if (orderInfo.length == 6) { int givenId = Integer.parseInt(orderInfo[1]); String dishName = orderInfo[2]; int level = Integer.parseInt(orderInfo[3]); int portion = Integer.parseInt(orderInfo[4]); int quantity = Integer.parseInt(orderInfo[5]); Dish dish = menu.searchDish(dishName); if (dish == null) { System.out.println(dishName + " does not exist"); orderLine = sc.nextLine(); continue; } Record record1 = new Record(givenId, orderNum, dish, level, portion, quantity); Record record2 = new Record(givenId, orderNum, dish, level, 0, quantity); tables.get(tableId - 1).addRecord(record1); tables.get(givenId - 1).addRecord(record2); if (legaltime) { System.out.println(record1.print(tableId)); } } else { System.out.println("wrong format"); } } } } // 读入下一个桌号标识 orderLine = sc.nextLine(); } sc.close(); for (Table table : tables) { if (table.flag && table.getTotalPrice() != 0) { System.out.println(table.printInfo()); } } names.sort(new Comparator<String>() { @Override public int compare(String s1, String s2) { return s1.compareTo(s2); } }); for (String costumName : names) { int sum = 0; String costumPhone = null; for (Table table : tables) { if (table.name.equals(costumName)) { sum += table.getCheckedPrice(); costumPhone = table.phone; } } if (sum != 0) { System.out.println(costumName + " " + costumPhone + " " + sum); } } } } enum DishType { Chuan, Jin, Zhe, } class Dish { public String name; public int unit_price; public DishType type; public Dish(String name, int unit_price, DishType type) { this.name = name; this.unit_price = unit_price; this.type = type; } public Dish(String name, int unit_price) { this.name = name; this.unit_price = unit_price; } @Override public String toString() { return name; } } class Menu { public ArrayList<Dish> dishs = new ArrayList<>(); public Dish searchDish(String dishName) { for (Dish dish : dishs) { if (dish.name.equals(dishName)) { return dish; } } return null; } void addDish(String dishName, int unit_price) { dishs.add(new Dish(dishName, unit_price)); } void addDish(String dishName, int unit_price, DishType type) { dishs.add(new Dish(dishName, unit_price, type)); } } class Record { int orderNum; Dish dish; int portion; int quantity; int level; boolean flag; int givenId; boolean check_level() { switch (dish.type) { case Chuan: if (level > 5 || level < 0) { return false; } else { return true; } case Jin: if (level > 4 || level < 0) { return false; } else { return true; } case Zhe: if (level > 3 || level < 0) { return false; } else { return true; } default: return true; } } public Record(int givenID, int orderNum, Dish dish, int portion, int quantity) { this.orderNum = orderNum; this.dish = dish; this.portion = portion; this.quantity = quantity; this.level = -1; this.flag = true; this.givenId = givenID; } public Record(int givenId, int orderNum, Dish dish, int level, int portion, int quantity) { this.orderNum = orderNum; this.dish = dish; this.portion = portion; this.quantity = quantity; this.level = level; this.flag = check_level(); this.givenId = givenId; } int getPrice() { if (!flag) return 0; double coefficient = 0; switch (portion) { case 1: coefficient = 1; break; case 2: coefficient = 1.5; break; case 3: coefficient = 2; break; } int price = (int) Math.round(dish.unit_price * coefficient) * quantity; return price; } int getCheckedPrice(Double coefficient) { return (int) Math.round(getPrice() * coefficient); } public String print(int tableId) { if (flag == false) { switch (dish.type) { case Chuan: return "spicy num out of range :" + level; case Jin: return "acidity num out of range :" + level; case Zhe: return "sweetness num out of range :" + level; default: return null; } } else { if (givenId == tableId) { return orderNum + " " + dish.toString() + " " + getPrice(); } return orderNum + " table " + tableId + " pay for table " + givenId + " " + getPrice(); } } @Override public String toString() { return "Record [orderNum=" + orderNum + ", dish=" + dish + ", portion=" + portion + ", quantity=" + quantity + ", level=" + level + ", flag=" + flag + ", givenId=" + givenId + "]"; } } class Table { ArrayList<Record> records = new ArrayList<>(); int tableId; String name; String phone; Date date; Date time; boolean flag; public Table(int tableId, String name, String phone, Date date, Date time) { this.name = name; this.phone = phone; this.date = date; this.time = time; this.tableId = tableId; this.flag = true; } double getCoefficient(boolean Special) throws ParseException { double coefficient = 0; SimpleDateFormat sdfTime = new SimpleDateFormat("HH:mm"); Calendar cal = Calendar.getInstance(); cal.setTime(date); int dayOfWeek = cal.get(Calendar.DAY_OF_WEEK); // 营业时间 if (dayOfWeek == 1 || dayOfWeek == 7) { if (time.after(sdfTime.parse("9:29")) && time.before(sdfTime.parse("21:31"))) { coefficient = 1; } } else { if (time.after(sdfTime.parse("16:59")) && time.before(sdfTime.parse("20:31"))) { if (Special) { coefficient = 0.7; } else { coefficient = 0.8; } } else if (time.after(sdfTime.parse("10:29")) && time.before(sdfTime.parse("14:31"))) { if (Special) { coefficient = 0.7; } else { coefficient = 0.6; } } } if (coefficient == 0) { flag = false; } return coefficient; } int getTotalPrice() { int sum = 0; for (Record record : records) { sum += record.getPrice(); } return sum; } int getCheckedPrice() throws ParseException { int sum = 0; for (Record record : records) { if (record.level != -1) { sum += record.getCheckedPrice(getCoefficient(true)); } else { sum += record.getCheckedPrice(getCoefficient(false)); } } return sum; } String getAveLevel(DishType type) { String[] spicy = { "不辣", "微辣", "稍辣", "辣", "很辣", "爆辣" }; String[] acidity = { "不酸", "微酸", "稍酸", "酸", "很酸" }; String[] sweetness = { "不甜", "微甜", "稍甜", "甜" }; double sum = 0; double num = 0; for (Record record : records) { if (record.dish.type == type) { if (record.flag && tableId == record.givenId) { num += record.quantity; sum += record.level * record.quantity; } } } if (num == 0) { return ""; } int ave = (int) Math.round(sum / num); switch (type) { case Chuan: return " 川菜 " + (int) num + " " + spicy[ave]; case Jin: return " 晋菜 " + (int) num + " " + acidity[ave]; case Zhe: return " 浙菜 " + (int) num + " " + sweetness[ave]; default: return null; } } void addRecord(Record record) { records.add(record); } boolean delRecordByOrderNum(int orderNum) { return records.removeIf(record -> record.orderNum == orderNum); } Record findRecordByOrderNum(int orderNum) { for (Record record : records) { if (record.orderNum == orderNum) { return record; } } return null; } public String printId() { return "table " + tableId + ": "; } public String printInfo() throws ParseException { String chuan = getAveLevel(DishType.Chuan); String jin = getAveLevel(DishType.Jin); String zhe = getAveLevel(DishType.Zhe); if (chuan == "" && jin == "" && zhe == "") { return "table " + tableId + ": " + getTotalPrice() + " " + getCheckedPrice() + " "; } else { return "table " + tableId + ": " + getTotalPrice() + " " + getCheckedPrice() + chuan + jin + zhe; } } }
我觉得菜单五要比菜单四简单一点点一点点而已
类图如下:

代码难度如下:

以下是我对代码的分析:
- 我先创建一个Menu对象来存储菜单信息。
- 通过Scanner对象从控制台读入菜单信息,按照特定的格式进行解析,将菜品信息添加到Menu对象中。
- 创建一个ArrayList来存储Table对象和一个ArrayList来存储顾客姓名。
- 通过Scanner对象从控制台读入订单信息,按照特定的格式进行解析,根据订单类型进行相应的操作。
- 将Table对象添加到Tables列表中,并检查订单的合法性和时间是否在营业时间范围内。
- 最后,根据需求打印输出订单信息和统计信息。
整个代码由多个类组成,每个类都有特定的功能:
- Dish类表示菜品,包含菜品名称、单价和菜品类型。
- Menu类表示菜单,包含一个菜品列表和相关的操作方法。
- Record类表示订单记录,包含订单编号、菜品、份数、数量、口味等信息。
- Table类表示餐桌,包含订单记录列表、餐桌号、顾客姓名、电话号码、日期、时间等信息,以及相关的操作方法。
代码还使用了枚举类型DishType来表示菜品类型(川菜、晋菜、浙菜)。
总体而言,该代码通过读取用户输入的菜单和订单信息,进行解析和处理,最后输出订单信息和统计信息。它实现了简单的餐厅订单管理功能。
三,期中考试
7-1 测验1-圆类设计 分数 12 作者 段喜龙 单位 南昌航空大学 创建一个圆形类(Circle),私有属性为圆的半径,从控制台输入圆的半径,输出圆的面积 输入格式: 输入圆的半径,取值范围为(0,+∞),输入数据非法,则程序输出Wrong Format,注意:只考虑从控制台输入数值的情况 输出格式: 输出圆的面积(保留两位小数,可以使用String.format(“%.2f”,输出数值)控制精度) 输入样例: 在这里给出一组输入。例如: 2.35 输出样例: 在这里给出相应的输出。例如: 17.35 代码长度限制 16 KB 时间限制 400 ms 内存限制 64 MB
我的代码是
import java.util.Scanner; public class Main{ public static void main(String[] args) { Scanner scanner = new Scanner(System.in); double radius = scanner.nextDouble(); Circle circle = new Circle(); circle.setRadius(radius); double area = circle.getArea(); if (area > 0) { String formattedArea = String.format("%.2f", area); System.out.println(formattedArea); } } } class Circle{ private double radius; public void setRadius(double radius) { if (radius > 0) { this.radius = radius; } else { System.out.println("Wrong Format"); } } public double getArea() { return Math.PI * radius * radius; } }
比较简单这题,就不多加分析了
7-2 测验2-类结构设计 分数 18 作者 段喜龙 单位 南昌航空大学 设计一个矩形类,其属性由矩形左上角坐标点(x1,y1)及右下角坐标点(x2,y2)组成,其中,坐标点属性包括该坐标点的X轴及Y轴的坐标值(实型数),求得该矩形的面积。类设计如下图: image.png 输入格式: 分别输入两个坐标点的坐标值x1,y1,x2,y2。 输出格式: 输出该矩形的面积值(保留两位小数)。 输入样例: 在这里给出一组输入。例如: 6 5.8 -7 8.9 输出样例: 在这里给出相应的输出。例如: 40.30 代码长度限制 16 KB 时间限制 400 ms 内存限制 64 MB
我的代码是
import java.util.Scanner; public class Main { public static void main(String[] args) { Scanner input = new Scanner(System.in); double x1 = input.nextDouble(); double y1 = input.nextDouble(); double x2 = input.nextDouble(); double y2 = input.nextDouble(); Point leftTopPoint = new Point(x1, y1); Point lowerRightPoint = new Point(x2, y2); Rectangle rectangle = new Rectangle(leftTopPoint, lowerRightPoint); System.out.printf("%.2f", rectangle.getArea()); } } class Rectangle { private Point leftTopPoint; private Point lowerRightPoint; public Rectangle(Point leftTopPoint, Point lowerRightPoint) { this.leftTopPoint = leftTopPoint; this.lowerRightPoint = lowerRightPoint; } public double getArea() { double length = Math.abs(lowerRightPoint.getX() - leftTopPoint.getX()); double width = Math.abs(lowerRightPoint.getY() - leftTopPoint.getY()); return length * width; } } class Point { private double x; private double y; public Point(double x, double y) { this.x = x; this.y = y; } public double getX() { return x; } public double getY() { return y; } }
其实说实话,这个也不是很难,就是在教室里考比较急,会紧张打错一些东西。
将测验1与测验2的类设计进行合并设计,抽象出Shape父类(抽象类),Circle及Rectangle作为子类,类图如下所示:

试编程完成如上类图设计,主方法源码如下(可直接拷贝使用):
public static void main(String[] args) {
// TODO Auto-generated method stub
Scanner input = new Scanner(System.in);
int choice = input.nextInt();
switch(choice) {
case 1://Circle
double radiums = input.nextDouble();
Shape circle = new Circle(radiums);
printArea(circle);
break;
case 2://Rectangle
double x1 = input.nextDouble();
double y1 = input.nextDouble();
double x2 = input.nextDouble();
double y2 = input.nextDouble();
Point leftTopPoint = new Point(x1,y1);
Point lowerRightPoint = new Point(x2,y2);
Rectangle rectangle = new Rectangle(leftTopPoint,lowerRightPoint);
printArea(rectangle);
break;
}
}
其中,printArea(Shape shape)方法为定义在Main类中的静态方法,体现程序设计的多态性。
输入格式:
输入类型选择(1或2,不考虑无效输入)
对应图形的参数(圆或矩形)
输出格式:
图形的面积(保留两位小数)
输入样例1:
1
5.6
输出样例1:
在这里给出相应的输出。例如:
98.52
输入样例2:
2
5.6
-32.5
9.4
-5.6
输出样例2:
在这里给出相应的输出。例如:
102.22
import java.util.Scanner; public class Main { public static void main(String[] args) { Scanner input = new Scanner(System.in); int choice = input.nextInt(); switch (choice) { case 1:// Circle double radius = input.nextDouble(); if (radius <= 0) { System.out.println("Wrong Format"); }else{ Shape circle = new Circle(radius); printArea(circle);} break; case 2:// Rectangle double x1 = input.nextDouble(); double y1 = input.nextDouble(); double x2 = input.nextDouble(); double y2 = input.nextDouble(); Point leftTopPoint = new Point(x1, y1); Point lowerRightPoint = new Point(x2, y2); Rectangle rectangle = new Rectangle(leftTopPoint, lowerRightPoint); printArea(rectangle); break; } input.close(); } public static void printArea(Shape shape) { System.out.printf("%.2f", shape.area()); } } abstract class Shape { public abstract double area(); } class Circle extends Shape { private double radius; public Circle(double radius) { this.radius = radius; } public double getRadius() { return radius; } public void setRadius(double radius) { this.radius = radius; } @Override public double area() { return Math.PI * radius * radius; } } class Rectangle extends Shape { private Point leftTopPoint; private Point lowerRightPoint; public Rectangle(Point leftTopPoint, Point lowerRightPoint) { this.leftTopPoint = leftTopPoint; this.lowerRightPoint = lowerRightPoint; } public Point getLeftTopPoint() { return leftTopPoint; } public void setLeftTopPoint(Point leftTopPoint) { this.leftTopPoint = leftTopPoint; } public Point getLowerRightPoint() { return lowerRightPoint; } public void setLowerRightPoint(Point lowerRightPoint) { this.lowerRightPoint = lowerRightPoint; } @Override public double area() { double length = Math.abs(lowerRightPoint.getX() - leftTopPoint.getX()); double width = Math.abs(lowerRightPoint.getY() - leftTopPoint.getY()); return length * width; } } class Point { private double x; private double y; public Point(double x, double y) { this.x = x; this.y = y; } public double getX() { return x; } public double getY() { return y; } }
在程序的main方法中,首先创建了一个Scanner对象input用于读取输入。然后通过input.nextInt()读取用户的选择并将其保存在choice变量中。
接下来,使用switch语句根据用户的选择执行不同的操作。如果choice等于1,表示用户选择了圆形,接着通过input.nextDouble()读取用户输入的半径并保存在radius变量中。然后根据半径创建一个Circle对象,并通过printArea方法打印出圆形的面积。
如果choice等于2,表示用户选择了矩形,接着通过input.nextDouble()读取用户输入的矩形的左上角和右下角坐标,并将其分别保存在x1、y1、x2和y2变量中。然后根据这些坐标创建一个Rectangle对象,并通过printArea方法打印出矩形的面积。
在程序的末尾,通过input.close()关闭了Scanner对象。
除了main方法之外,还定义了其他类和方法。Shape是一个抽象类,其中定义了一个抽象方法area(),表示计算形状的面积。Circle和Rectangle类分别继承了Shape类,并实现了area()方法来计算圆形和矩形的面积。
另外,还定义了一个Point类,用于表示坐标点。Circle类和Rectangle类中分别使用了Point对象来表示圆心和矩形的两个顶点。

在测验3的题目基础上,重构类设计,实现列表内图形的排序功能(按照图形的面积进行排序)。
提示:题目中Shape类要实现Comparable接口。
其中,Main类源码如下(可直接拷贝使用):
public class Main {
public static void main(String\[\] args) {
// TODO Auto-generated method stub
Scanner input = new Scanner(System.in);
ArrayList<Shape> list = new ArrayList<>();
int choice = input.nextInt();
while(choice != 0) {
switch(choice) {
case 1://Circle
double radiums = input.nextDouble();
Shape circle = new Circle(radiums);
list.add(circle);
break;
case 2://Rectangle
double x1 = input.nextDouble();
double y1 = input.nextDouble();
double x2 = input.nextDouble();
double y2 = input.nextDouble();
Point leftTopPoint = new Point(x1,y1);
Point lowerRightPoint = new Point(x2,y2);
Rectangle rectangle = new Rectangle(leftTopPoint,lowerRightPoint);
list.add(rectangle);
break;
}
choice = input.nextInt();
}
list.sort(Comparator.naturalOrder());//正向排序
for(int i = 0; i < list.size(); i++) {
System.out.print(String.format("%.2f", list.get(i).getArea()) + " ");
}
}
}
输入格式:
输入图形类型(1:圆形;2:矩形;0:结束输入)
输入图形所需参数
输出格式:
按升序排序输出列表中各图形的面积(保留两位小数),各图形面积之间用空格分隔。
输入样例:
在这里给出一组输入。例如:
1
2.3
2
3.2
3
6
5
1
2.3
0
输出样例:
在这里给出相应的输出。例如:
5.60 16.62 16.62
import java.util.ArrayList; import java.util.Comparator; import java.util.Scanner; public class Main { public static void main(String[] args) { Scanner input = new Scanner(System.in); ArrayList<Shape> list = new ArrayList<>(); int choice = input.nextInt(); while (choice != 0) { switch (choice) { case 1: double radius = input.nextDouble(); if (radius <= 0) { System.out.println("Wrong Format"); }else{ Circle circle = new Circle(radius); list.add(circle);} break; case 2: double x1 = input.nextDouble(); double y1 = input.nextDouble(); double x2 = input.nextDouble(); double y2 = input.nextDouble(); Point leftTopPoint = new Point(x1, y1); Point lowerRightPoint = new Point(x2, y2); Rectangle rectangle = new Rectangle(leftTopPoint, lowerRightPoint); list.add(rectangle); break; } choice = input.nextInt(); } list.sort(Comparator.naturalOrder()); for (int i = 0; i < list.size(); i++) { System.out.print(String.format("%.2f", list.get(i).getArea()) + " "); } } } class Point { private double x; private double y; public Point(double x, double y) { this.x = x; this.y = y; } public double getX() { return x; } public double getY() { return y; } } class Rectangle extends Shape { private Point leftTopPoint; private Point lowerRightPoint; public Rectangle(Point leftTopPoint, Point lowerRightPoint) { this.leftTopPoint = leftTopPoint; this.lowerRightPoint = lowerRightPoint; } public Point getLeftTopPoint() { return leftTopPoint; } public Point getLowerRightPoint() { return lowerRightPoint; } @Override public double getArea() { double width = lowerRightPoint.getX() - leftTopPoint.getX(); double height = leftTopPoint.getY() - lowerRightPoint.getY(); return width * height; } } class Circle extends Shape { private double radius; public Circle(double radius) { this.radius = radius; } public double getRadius() { return radius; } @Override public double getArea() { return Math.PI * radius * radius; } } abstract class Shape implements Comparable<Shape>{ public abstract double getArea(); @Override public int compareTo(Shape o) { if (this.getArea() < o.getArea()) { return -1; } else if (this.getArea() > o.getArea()) { return 1; } else { return 0; } } }
期中考试就是在一个题目上不断迭代,从最基础的加上继承和多态,抽象类和接口。
代码难度:

- 用户可以输入圆形和矩形的信息,并将它们保存到一个ArrayList集合中。
- 用户可以连续输入多个形状信息,直到输入0为止。
- 根据用户输入的形状信息,创建对应的Circle和Rectangle对象,并将它们添加到ArrayList集合中。
- 最后,对ArrayList集合中的形状对象按照面积大小进行排序,并将排序后的结果输出。
我这个代码中使用了面向对象的思想,定义了抽象类Shape作为所有形状的父类,Circle和Rectangle类继承自Shape类,并实现了各自的getArea方法来计算形状的面积。Shape类还实现了Comparable接口,使得形状对象可以进行比较和排序。

(3)采坑心得:
在菜单四中我遇到的一些问题和踩坑心得:
-
输入格式验证:确保输入的格式符合要求,例如正确的桌号格式、时间格式以及菜单、订单的格式。对于不合法的输入,需要进行适当的错误处理。
-
数据结构选择:选择合适的数据结构来存储菜谱、订单和点菜记录等信息。在这个问题中,使用哈希表(HashMap、TreeMap)可以方便地进行快速查找和存储。
-
菜谱信息与订单信息分离:在处理输入时,需要将菜谱信息和订单信息分开处理,并进行合适的操作。菜谱信息应添加到菜单对象中,订单信息应根据桌号查找或创建订单对象。
-
删除记录和合并记录:在处理删除记录时,需要注意删除的合法性和重复删除的情况。另外,如果有相同桌号、菜名和份额的点菜记录,应将它们合并为一条记录进行计算,避免四舍五入误差的累积。
-
营业时间和折扣计算:根据问题描述中提供的营业时间和折扣规则,需要编写逻辑来判断是否在营业时间内,并计算相应的折扣。
-
异常情况处理:在代码中应该考虑各种异常情况,并进行适当的错误处理。例如,处理不存在的桌号、重复的菜谱信息、重复的删除记录等情况,给出合适的错误提示。
在菜单五中我遇到的一些问题和踩坑心得:
-
访问修饰符错误:忘记添加正确的访问修饰符(如
public、private)可能导致无法访问或访问权限错误。 -
参数传递错误:在调用方法或构造函数时,传递的参数类型、顺序或数量与方法定义不匹配,这会导致编译错误或运行时错误。
-
空指针异常:使用未初始化的对象或空引用调用方法或访问属性会导致空指针异常。在使用对象之前,要确保其已经被正确地初始化。
-
下标越界:在访问数组、集合或字符串时,使用超出其索引范围的下标会导致下标越界异常。要确保使用有效的索引。
-
对象比较错误:使用
==比较对象时,比较的是对象的引用而不是内容。要比较对象的内容,应使用equals()方法。 -
循环条件错误:在编写循环时,循环条件的判断错误可能导致死循环或无法执行循环体。
在期中考试中我遇到的一些问题和踩坑心得:
1.参数错误,使用构造方法时需要注意参数的类型和顺序是否正确,否则会出现编译错误或运行时错误。
2.循环的范围

当时就是吧break也包括进去了。
(4)改进建议:
期中:
1.类的封装:考虑将每个类放在单独的文件中,并使用适当的访问修饰符(如private、protected、public)来限制成员变量和方法的访问权
2.操作符的连续使用:在使用Scanner读取输入时,可以考虑使用操作符的连续使用来简化代码。例如,可以将以下代码段:
double x1 = input.nextDouble();
double y1 = input.nextDouble();
double x2 = input.nextDouble();
double y2 = input.nextDouble();
简化为
double x1, y1, x2, y2;
x1 = input.nextDouble();
y1 = input.nextDouble();
x2 = input.nextDouble();
y2 = input.nextDouble();
菜单系类:
- 在使用try-catch块捕获异常时,尽量避免捕获所有异常,可以只捕获需要处理的特定异常类型,以便更好地处理异常情况。
- 在比较字符串时,推荐使用
equals()方法而不是"=="运算符,以确保比较的是字符串的内容而不是引用。 - 考虑使用更简洁的语法来初始化集合。例如,可以使用更现代的方式初始化HashMap,如
Map<String, String> map = Map.of("川菜", "Chuan", "晋菜", "Jin", "浙菜", "Zhe");。
4.重写toString()方法:在Shape、Circle和Rectangle类中,可以重写toString()方法以提供更有意义的字符串表示形式。这将有助于调试和输出对象时的可读性
5. 将 Record 类中的 flag 属性改为 isValid 或 isFlagged,更能表达其含义。
6.将 Record 类中的 givenId 属性改为 givenTableId,以更清晰地表示其含义。
7.考虑将 Record 类的一些逻辑(如价格计算和级别检查)封装为方法,以提高代码的可读性和可维护性。
(5)总结:
通过编写这个菜单系类,我学会了如何使用Java编程语言来构建一个基本的命令行应用程序,了解了面向对象编程的基本概念和原则,并将其应用到了项目中。
学习了类的定义和使用,对象的创建和操作,以及如何设计和组织代码结构。掌握了异常处理的方法和技巧,以及如何在程序中有效地处理错误和异常情况。
学会了使用集合类(如ArrayList和HashMap)来存储和管理数据,并应用它们来实现系统的各项功能。通过自己的思考和查阅相关资料,我成功地解决了一些问题
并提高了对Java编程的理解能力。一步一步逐渐完善自己设计,这是个艰辛但成果很愉悦的过程。还有就是编程思维得到了训练和提升,掌握了一些基础题目的编
思想。了解了开闭原则,及继承和多态的使用,对面向对象又有了一层新的认识。在写PTA的过程中我意识到自己代码的可读性十分差,条理性还不够清晰,我应
注重代码书写习惯。所以学习JAVA,不仅对我们以后学习其他语言有很大的好处,而且也让我们知道了和理解了作为一个编程人员首先应具有的良好心理素质,那
是冷静思考和专心致志。对待学术知识应该是严谨和认真。
路漫漫其修远兮。