作业4,5,6以及期中考试
一丶前言:
这三次作业都有一个菜单计价程序,虽然题目量不大,但是三个菜单计价程序难度都很高,需要很长时间才能做出来,并且不一定能够拿到及格分,所以总的来说作业难度还是很高,期中考试体量不大,并且难度也不高,可能是因为没有多少时间,所以老师给我们出的题目难度也并不是很高
二丶设计与分析:
菜单计价程序-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+英文空格+桌号+“:”+英文空格+当前桌的总价
本次题目不考虑其他错误情况,如:桌号、菜单订单顺序颠倒、不符合格式的输入、序号重复等,在本系列的后续作业中会做要求。
输入格式:
桌号标识格式: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
table 1 2023/3/22 12/2/3
1 麻婆豆腐 2 2
2 油淋生菜 1 3
end
输出样例:
在这里给出相应的输出。例如:
table 1:
1 麻婆豆腐 36
2 油淋生菜 27
table 1: 38
输入样例1:
在这里给出一组输入。例如:
麻婆豆腐 12
油淋生菜 9
table 1 2023/3/22 17/0/0
1 麻婆豆腐 2 2
2 油淋生菜 1 3
1 delete
end
输出样例1:
在这里给出相应的输出。例如:
table 1:
1 麻婆豆腐 36
2 油淋生菜 27
table 1: 22
输入样例2:
在这里给出一组输入。例如:
麻婆豆腐 12
油淋生菜 9
table 1 2023/3/22 16/59/59
1 麻婆豆腐 2 2
2 油淋生菜 1 3
1 delete
end
输出样例2:
在这里给出相应的输出。例如:
table 1:
1 麻婆豆腐 36
2 油淋生菜 27
table 1 out of opening hours
输入样例3:
在这里给出一组输入。例如:
麻婆豆腐 12
油淋生菜 9
table 1 2022/12/5 15/03/02
1 麻婆豆腐 2 2
2 油淋生菜 1 3
3 麻辣鸡丝 1 2
5 delete
7 delete
table 2 2022/12/3 15/03/02
1 麻婆豆腐 2 2
2 油淋生菜 1 3
3 麻辣鸡丝 1 2
7 delete
end
输出样例3:
在这里给出相应的输出。例如:
table 1:
1 麻婆豆腐 36
2 油淋生菜 27
麻辣鸡丝 does not exist
delete error;
delete error;
table 2:
1 麻婆豆腐 36
2 油淋生菜 27
麻辣鸡丝 does not exist
delete error;
table 1 out of opening hours
table 2: 63
输入样例4:
在这里给出一组输入。例如:
麻婆豆腐 12
油淋生菜 9
table 1 2022/12/3 19/5/12
1 麻婆豆腐 2 2
2 油淋生菜 1 3
3 麻辣鸡丝 1 2
table 2 2022/12/3 15/03/02
1 麻婆豆腐 2 2
2 油淋生菜 1 3
3 麻辣鸡丝 1 2
1 4 麻婆豆腐 1 1
7 delete
end
输出样例4:
在这里给出相应的输出。例如:
table 1:
1 麻婆豆腐 36
2 油淋生菜 27
麻辣鸡丝 does not exist
table 2:
1 麻婆豆腐 36
2 油淋生菜 27
麻辣鸡丝 does not exist
4 table 2 pay for table 1 12
delete error;
table 1: 63
table 2: 75
import java.util.HashMap;
import java.util.Map;
import java.util.Scanner;
public class Main {
public static void main(String[] args) {
Map<Integer, Table> tables = new HashMap<>();
Scanner scanner = new Scanner(System.in);
String line;
while (!(line = scanner.nextLine()).equals("end")) {
String[] parts = line.split(" ");
if (parts.length == 2) {
String dishName = parts[0];
int score = Integer.parseInt(parts[1]);
for (Table table : tables.values()) {
if (table.containsDish(dishName)) {
table.addScore(dishName, score);
}
}
} else if (parts.length == 4 && parts[0].equals("table")) {
int tableNumber = Integer.parseInt(parts[1]);
String date = parts[2];
String time = parts[3];
Table table = new Table(tableNumber, date, time);
tables.put(tableNumber, table);
}
}
for (Table table : tables.values()) {
System.out.println(table);
}
}
static class Table {
private int number;
private String date;
private String time;
private Map<String, Integer> scores;
public Table(int number, String date, String time) {
this.number = number;
this.date = date;
this.time = time;
this.scores = new HashMap<>();
}
public void addScore(String dishName, int score) {
scores.put(dishName, scores.getOrDefault(dishName, 0) + score);
}
public boolean containsDish(String dishName) {
return scores.containsKey(dishName);
}
public int getTotalScore() {
int totalScore = 0;
for (int score : scores.values()) {
totalScore += score;
}
return totalScore;
}
@Override
public String toString() {
StringBuilder sb = new StringBuilder();
sb.append("table ").append(number).append(":").append("\n");
for (Map.Entry<String, Integer> entry : scores.entrySet()) {
sb.append(entry.getKey()).append(" ").append(entry.getValue()).append("\n");
}
sb.append("table ").append(number).append(": ").append(getTotalScore());
return sb.toString();
}
}
}
设计与分析:
-
数据结构设计:
- 程序中定义了一个内部类 Table,用于表示餐桌的信息,包括编号、日期、时间以及点菜情况和评分。
- 使用 Map<Integer, Table> tables 来存储所有餐桌的信息,其中键为餐桌编号,值为对应的 Table 对象。
-
输入处理:
- 通过 Scanner 从标准输入读取用户输入的信息,根据输入的不同情况进行相应的处理。
- 如果输入是菜品名和评分,就遍历所有餐桌,找到含有该菜品的餐桌并更新评分;如果输入是新开台的信息,则创建新的 Table 对象并放入 tables 中。
-
Table 类方法设计:
- addScore 方法用于给指定菜品添加评分,若菜品已有评分则累加,否则创建新的评分项。
- containsDish 方法用于判断该餐桌是否点过某道菜。
- getTotalScore 方法用于计算该餐桌所有菜品的总评分。
- toString 方法用于将餐桌的信息转换为字符串,包括餐桌号、各菜品评分以及总评分。
-
输出结果:
- 最后通过迭代输出所有餐桌的信息,包括点菜情况和总评分。

菜单计价程序-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
import java.util.Scanner;
import java.text.DateFormat;
import java.util.ArrayList;
import java.util.Calendar;
import java.util.Date;
import java.text.SimpleDateFormat;
import java.text.ParseException;
public class Main {
public static void main(String[] args) {
Main main = new Main();
Scanner scanner = new Scanner(System.in);
ArrayList<String> messageList = new ArrayList<String>();
ArrayList<String> arrayList = new ArrayList<String>();
ArrayList<Table> tArrayList = new ArrayList<Main.Table>();
String string = "";
while (true) {
string = scanner.nextLine();
if ("end".equals(string)) {
break;
}
arrayList.add(string);
}
Menu menu = main.new Menu();
menu.dishs = new Dish[0];
int n = 0;
int num1=0;
while (num1< arrayList.size()) {
String[] strings = arrayList.get(num1).split(" ");
if (strings.length==0) {
messageList.add("wrong format");
break;
}
if (strings.length > 3) {
if(strings[0].charAt(0)=='t')
{
n = num1;
break;
}
if(strings[0].charAt(0)!='t'||strings[0].charAt(0)=='t')
messageList.add("wrong format");
continue;
}
String dPrice = strings[1];
if (dPrice.charAt(0) == '0') {
messageList.add("wrong format");
continue;
}
String dName = strings[0];
int hp = 0;
try {
hp = Integer.parseInt(dPrice);
} catch (Exception e) {
messageList.add("wrong format");
continue;
}
if ( hp <= 0) {
messageList.add(dName + " price out of range "+hp);
continue;
}
if (hp >= 300 ) {
messageList.add(dName + " price out of range "+hp);
continue;
}
if (menu.searthDish(dName) != null) {
int num2=0;
while (num2 < menu.dishs.length) {
if (menu.dishs[num2].equals(dName)) {//对比菜单名称
menu.dishs[num2].unit_price = hp;
}
num2++;
}
} else {
Dish dish = main.new Dish();
if (strings.length == 3&&strings.length != 2) {
dish.name = dName;
dish.unit_price = hp;
dish.isT = true;
}
else if (strings.length == 2&&strings.length != 3) {
dish.name = dName;
dish.unit_price = hp;
}
menu.addDish(dish);
}
num1++;
}
int num3=0;
ArrayList<Integer> arrayLista = new ArrayList<Integer>();// 桌号开始下标
while(num3 < arrayList.size()) {
if (arrayList.get(num3).split(" ")[0].length() > 2 && arrayList.get(num3).split(" ")[0].charAt(0) == 't') {
arrayLista.add(num3);
}
num3++;
}
int num4=0;
for (int i = 0; i < arrayLista.size(); i++){
Order order = main.new Order();
Table table = main.new Table();
order.records = new Record[0];
table.order = order;
int t1 = arrayLista.get(i);
String[] strings = arrayList.get(t1).split(" ");
if (strings.length==0) {
messageList.add("wrong format");
continue;
}
if (strings[0].equals("table")) {
if (strings[1].charAt(0) == '0') {
messageList.add("wrong format");
continue;
}
int tNum = 0;
try {
tNum = Integer.parseInt(strings[1]);
} catch (Exception e) {
messageList.add("wrong format");
continue;
}
if (tNum > 55 )
if(tNum <= 0){
messageList.add(tNum + " num out of range");
continue;
}
if (strings[2].split("/")[0].length() != 4) {// year
messageList.add("wrong format");
continue;
}
if (strings[2].split("/")[1].length() > 2 || strings[2].split("/")[1].length() < 1) {// month
messageList.add("wrong format");
continue;
}
if (strings[2].split("/")[2].length() > 2 || strings[2].split("/")[2].length() < 1) {// day
messageList.add("wrong format");
continue;
}
if (strings[3].split("/")[0].length() > 2 || strings[3].split("/")[0].length() < 1) {// hour
messageList.add("wrong format");
continue;
}
if (strings[3].split("/")[1].length() > 2 || strings[3].split("/")[1].length() < 1) {// minute
messageList.add("wrong format");
continue;
}
if (strings[3].split("/")[2].length() > 2 || strings[3].split("/")[2].length() < 1) {// moment
messageList.add("wrong format");
continue;
}
DateFormat sdf = new SimpleDateFormat("yyyy/MM/dd HH/mm/ss");
sdf.setLenient(false);
Date date = null;
try {
date = sdf.parse(strings[2] + " " + strings[3]);
} catch (ParseException e) {
messageList.add(tNum + " date error");
continue;
}
table.date = date;
table.tableNum = tNum;
SimpleDateFormat sdf1 = new SimpleDateFormat("yyyy/MM/dd");
SimpleDateFormat sdf2 = new SimpleDateFormat("HH/mm/ss");
Date date1 = null;
Date date2 = null;
try {
date1 = sdf1.parse("2022/1/1");
date2 = sdf1.parse("2023/12/31");
if (date1.compareTo(sdf1.parse(strings[2])) > 0 || date2.compareTo(sdf1.parse(strings[2])) < 0) {
messageList.add("not a valid time period");
continue;
}
} catch (ParseException e1) {
// TODO Auto-generated catch block
e1.printStackTrace();
}
Calendar calendar = Calendar.getInstance();
calendar.setTime(date);
int dayNum = calendar.get(Calendar.DAY_OF_WEEK) - 1;
Date bdate = null;
Date edate = null;
Date bdate1 = null;
Date edate1 = null;
Date bdate2 = null;
Date edate2 = null;
Date nowDate = null;
try {
bdate = sdf2.parse("9/29/59");
edate = sdf2.parse("21/30/00");
bdate1 = sdf2.parse("10/30/00");
edate1 = sdf2.parse("14/30/00");
bdate2 = sdf2.parse("16/59/59");
edate2 = sdf2.parse("20/30/00");
nowDate = sdf2.parse(strings[3]);
} catch (ParseException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
if (dayNum == 6 || dayNum == 0) {// 周末
if (timeCalendar(nowDate, bdate, edate)) {// 全价
table.dicount = 1.0;
} else {
messageList.add("table " + tNum + " out of opening hours");
continue;
}
} if (dayNum != 6 && dayNum != 0){// 工作日
if (timeCalendar(nowDate, bdate1, edate1)) {// 6折
table.dicount = 0.6;
} else if (timeCalendar(nowDate, bdate2, edate2)) {// 8折
table.dicount = 0.8;
} else {
messageList.add("table " + tNum + " out of opening hours");
continue;
}
}
messageList.add("table"+" "+tNum + ": ");
if (t1 == arrayList.size()-1) {
} else {
int fNum;
if (arrayLista.get(arrayLista.size()-1) == t1) {
fNum = arrayList.size();
} else {
fNum = arrayLista.get(i + 1);
}
ArrayList<Integer> delNumArrayList = new ArrayList<Integer>();
int numb = 0;
for (int j = t1 + 1; j < fNum; j++) {// 点菜记录
String[] strings2 = arrayList.get(j).split(" ");
if (strings2.length==0) {
messageList.add("wrong format");
continue;
}
int Num1;
try {
Num1 = Integer.parseInt(strings2[0]);
} catch (Exception e) {
messageList.add("invalid dish");
continue;
}
if (strings2[0].charAt(0)=='0') {
messageList.add("wrong format");
continue;
}
if (strings2.length == 5) {// 代点菜
boolean flag = false;
if (tArrayList.get(0).tableNum == Num1)
flag = true;
if (flag) {
messageList.add("Table number :" + Num1 + " does not exist");
continue;
}
if (strings2[3].length() > 1) {
messageList.add("wrong format");
continue;
} else {
String dishName = strings2[2];// 菜名
int dishP;
int dishN;
if (strings2[4].charAt(0) == '0') {
messageList.add("wrong format");
continue;
}
try {
dishP = Integer.parseInt(strings2[3]);
dishN = Integer.parseInt(strings2[4]);
} catch (Exception e) {
messageList.add("wrong format");
continue;
}
if (menu.searthDish(dishName) == null) {
messageList.add(dishName + " does not exist");
continue;
}
Dish dish = menu.searthDish(dishName);
if (dish.isT&&dishP != 1 ) {
if ( dishP != 3&& dishP != 2) {
messageList.add(Num1 + " portion out of range " + dishP);
continue;
}
} if (!dish.isT&&dishP != 3) {
if (dishP != 1 && dishP != 2) {
messageList.add(Num1 + " portion out of range " + dishP);
continue;
}
}
if ( dishN < 1) {
messageList.add(Num1 + " num out of range " + dishN);
continue;
}
if (dishN > 15 ) {
messageList.add(Num1 + " num out of range " + dishN);
continue;
}
boolean f1 = true;
if (f1) {
table.order.addARecord(Num1, dish, dishP, dishN);
}
}
} else if (strings2.length == 2) {
if (!strings2[1].equals("delete")) {
messageList.add("wrong format");
continue;
} else {
boolean flag = false;
for (Integer n2 : delNumArrayList) {
if (n2 == Num1) {
flag = true;
}
}
if (flag) {
messageList.add("deduplication " + Num1);
continue;
}
if(0!=0)
continue;
if (table.order.findRecordByNum(Num1) == -1) {
messageList.add("delete error");
continue;
}
table.order.delARecordByOrderNum(Num1);
if(0!=0)
continue;
delNumArrayList.add(Num1);
}
} else if (strings2.length == 4) {
if (strings2[2].length() > 1) {
messageList.add("wrong format");
continue;
} if (strings2[2].length() <=1) {
String dishName = strings2[1];// 菜名
int dishP;
int dishN;
if (strings2[3].charAt(0) == '0') {
messageList.add("wrong format");
continue;
}
try {
dishP = Integer.parseInt(strings2[2]);
int muniu1=0;
muniu1=muniu1+1;
dishN = Integer.parseInt(strings2[3]);
} catch (Exception e) {
messageList.add("wrong format");
continue;
}
if (Num1 <= numb) {
messageList.add("record serial number sequence error");
continue;
}
if (menu.searthDish(dishName) == null) {
messageList.add(dishName + " does not exist");
continue;
}
Dish dish = menu.searthDish(dishName);
if (dishP != 3&& dishP != 2) {
if (dishP != 1 &&dish.isT) {
messageList.add(Num1 + " portion out of range " + dishP);
continue;
}
} if (!dish.isT&&dishP != 2) {
if (dishP != 1 && dishP != 3 ) {
messageList.add(Num1 + " portion out of range " + dishP);
continue;
}
}
if (dishN > 15 || dishN < 1) {
messageList.add(Num1 + " num out of range " + dishN);
continue;
}
numb = Num1;
boolean f1 = true;
int k=0;
while(k < table.order.records.length) {
if (table.order.records[k].d.name.equals(dishName)
&& table.order.records[k].portion == dishP) {
table.order.records[k].num = table.order.records[k].num + dishN;
f1 = false;
}
k++;
}
if (f1) {
table.order.addARecord(Num1, dish, dishP, dishN);
messageList.add(Num1 + " " + dishName + " " + dish.getPrice(dishP) * dishN);
}
}
} else {
messageList.add("wrong format");
continue;
}
}
}
} else {
messageList.add("wrong format");
if (t1 != arrayList.size()-1)
{
int fNum=0;
if (arrayLista.get(arrayLista.size() - 1) != t1) {
fNum = arrayLista.get(i + 1)+1;
} if (arrayLista.get(arrayLista.size() - 1) == t1){
fNum = arrayList.size();
}
Table table2 = tArrayList.get(tArrayList.size()-1);
ArrayList<Integer> delNumArrayList = new ArrayList<Integer>();
int numb = table2.order.records[table2.order.records.length-1].orderNum;
for (int j = t1 + 1; j < fNum; j++) {// 点菜记录
String[] strings2 = arrayList.get(j).split(" ");
if (strings2.length==0) {
messageList.add("wrong format");
continue;
}
int Num1;
try {
Num1 = Integer.parseInt(strings2[0]);
} catch (Exception e) {
messageList.add("invalid dish");
continue;
}
if (strings2[0].charAt(0)=='0') {
messageList.add("wrong format");
continue;
}
if (strings2.length == 5) {// 代点菜
boolean flag = true;
int num20=0;
while (num20< tArrayList.size()) {
if (tArrayList.get(num20).tableNum == Num1) {
flag = false;
}
num20++;
}
if (strings2[3].length() > 1) {
messageList.add("wrong format");
continue;
}
if (flag) {
messageList.add("Table number :" + Num1 + " does not exist");
continue;
}if (strings2[3].length() <=1) {
String dishName = strings2[2];// 菜名
int dishP;
int dishN;
if (strings2[4].charAt(0) == '0') {
messageList.add("wrong format");
continue;
}
try {
dishP = Integer.parseInt(strings2[3]);
dishN = Integer.parseInt(strings2[4]);
} catch (Exception e) {
messageList.add("wrong format");
continue;
}
if (menu.searthDish(dishName) == null) {
messageList.add(dishName + " does not exist");
continue;
}
Dish dish = menu.searthDish(dishName);
if (dishP != 1&&dish.isT)
if(dishP != 3){
messageList.add(Num1 + " portion out of range " + dishP);
continue;
}
if (!dish.isT&&dishP != 3) {
if (dishP != 1 && dishP != 2) {
messageList.add(Num1 + " portion out of range " + dishP);
continue;
}
}
if (dishN > 15 ) {
messageList.add(Num1 + " num out of range " + dishN);
continue;
}
if (dishN < 1) {
messageList.add(Num1 + " num out of range " + dishN);
continue;
}
boolean f1 = true;
if (f1) {
table2.order.addARecord(Num1, dish, dishP, dishN);
}
}
} else if (strings2.length == 2) {
if (!strings2[1].equals("delete")) {
messageList.add("wrong format");
continue;
} else {
boolean flag = false;
for (Integer n2 : delNumArrayList) {
if (n2 == Num1) {
flag = true;
}
}
if (flag) {
messageList.add("deduplication " + Num1);
continue;
}
if (table2.order.findRecordByNum(Num1) == -1) {
messageList.add("delete error");
continue;
}
table2.order.delARecordByOrderNum(Num1);
delNumArrayList.add(Num1);
}
} else if (strings2.length == 4) {
if (strings2[2].length() > 1) {
messageList.add("wrong format");
continue;
} else {
String dishName = strings2[1];// 菜名
int dishP;
int dishN;
if (strings2[3].charAt(0) == '0') {
messageList.add("wrong format");
continue;
}
try {
dishP = Integer.parseInt(strings2[2]);
int muniu=0;
muniu=muniu+1;
dishN = Integer.parseInt(strings2[3]);
} catch (Exception e) {
messageList.add("wrong format");
continue;
}
if (Num1 <= numb) {
messageList.add("record serial number sequence error");
continue;
}
if (menu.searthDish(dishName) == null) {
messageList.add(dishName + " does not exist");
continue;
}
Dish dish = menu.searthDish(dishName);
if (dishP != 3&&dish.isT)
if(dishP != 1){
messageList.add(Num1 + " portion out of range " + dishP);
continue;
}
if(!dish.isT){
if (dishP != 1 && dishP != 3 )
if(dishP != 2){
messageList.add(Num1 + " portion out of range " + dishP);
continue;
}
}
if (dishN > 15 || dishN < 1) {
messageList.add(Num1 + " num out of range " + dishN);
continue;
}
numb = Num1;
boolean f1 = true;
int num11=0;
while(num11< table2.order.records.length) {
if (table2.order.records[num11].d.name.equals(dishName)
&& table2.order.records[num11].portion == dishP) {
table2.order.records[num11].num = table2.order.records[num11].num + dishN;
f1 = false;
}
num11++;
}
if (f1) {
table2.order.addARecord(Num1, dish, dishP, dishN);
messageList.add(Num1 + " "+dishName+" " + dish.getPrice(dishP) *dishN);
}
}
} else {
messageList.add("wrong format");
continue;
}
}
}
}
if (table.tableNum!=0) {
tArrayList.add(table);
}
}
int num6=0;
while (num6 < messageList.size()) {
System.out.println(messageList.get(num6));
num6++;
}
int num7=0;
while(num7< tArrayList.size()-1) {
if (tArrayList.get(0).tableNum<tArrayList.get(1).tableNum) {
Table table = main.new Table();
table = tArrayList.get(1);
tArrayList.set(1, tArrayList.get(0));
tArrayList.set(0, table);
}if (tArrayList.get(0).tableNum>=tArrayList.get(1).tableNum)
num7++;
}
int num8=0;
while(num8< tArrayList.size()) {
int tableNum = tArrayList.get(num8).tableNum;
Order order =tArrayList.get(num8).order;
int sum = 0;
int sumP = 0;
int num9=0;
while(num9< order.records.length) {
Record record = order.records[num9];
if (!record.d.isT||tArrayList.get(num8).dicount==1.0) {
sum +=record.getPrice();
sumP +=(int)Math.round(record.getPrice()*tArrayList.get(num8).dicount);
}if (record.d.isT&&tArrayList.get(num8).dicount!=1.0){
sum +=(int)Math.round(record.getPrice());
sumP +=(int)Math.round(record.getPrice()*0.7);
}
num9++;
}
if (num8!=tArrayList.size()-1) {
System.out.println("table"+" "+tableNum+":"+" "+sum+" "+sumP);
}
if (num8==tArrayList.size()-1) {
System.out.print("table"+" "+tableNum+":"+" "+sum+" "+sumP);
}
num8++;
}
}
// 菜品类
class Dish {
String name;// 菜品名称
int unit_price; // 单价
boolean isT = false;
int getPrice(int portion) {// 计算菜品价格的方法,输入参数是点菜的份额(输入数据只能是1/2/3,代表小/中/大份)
double money = 0;
if(portion==1)
money = this.unit_price * 1.0;
if(portion==2)
money = 1.5 * this.unit_price;
if(portion==3)
money = this.unit_price * 2.0;
return (int) Math.round(money);
}
}
// 菜谱类:对应菜谱,包含饭店提供的所有菜的信息
class Menu {
Dish[] dishs;// 菜品数组,保存所有菜品信息
Dish searthDish(String dishName) {
int num=0;
while(num < dishs.length) {
if (dishs[num].name.equals(dishName)) {
return dishs[num];
}
num++;
}
return null;
}
Dish addDish(Dish dish) {// 添加一道菜品信息
Dish[] dishs1 = new Dish[dishs.length + 1];
int num=0;
while(num < dishs.length) {
dishs1[num] = dishs[num];
num++;
}
dishs1[dishs.length] = dish;
dishs = dishs1;
return dish;
}
}
class Record {
int orderNum;
Dish d;// 菜品
int portion;
int num;// 份数
int getPrice() {// 计价,计算本条记录的价格
return (int) Math.round(this.d.getPrice(this.portion) * num);
}
}
// 订单类:保存用户点的所有菜的信息
class Order {
Record[] records;// 保存订单上每一道的记录
int getTotalPrice() {// 计算订单的总价
int sum = 0;
int num=0;
while (num < records.length) {
sum += records[num].getPrice();
num++;
}
return sum;
}
// 添加一条菜品信息到订单中
Record addARecord(int orderNum, Dish dish, int portion, int num) {
Record record = new Record();
record.orderNum = orderNum;
record.d = dish;
record.portion = portion;
record.num = num;
Record[] records1 = new Record[this.records.length + 1];
for (int i = 0; i < this.records.length; i++) {
records1[i] = this.records[i];
}
records1[this.records.length] = record;
this.records = records1;
return record;
}
// 根据序号删除一条记录
void delARecordByOrderNum(int orderNum) {
int n = this.findRecordByNum(orderNum);
Record[] records1 = new Record[this.records.length - 1];
for (int i = 0; i < this.records.length; i++) {
if (i == n) {
continue;
}
if (i > n) {
records1[i] = this.records[i - 1];
} else {
records1[i] = this.records[i];
}
}
this.records = records1;
}
// 根据序号查找一条记录
int findRecordByNum(int orderNum) {
int n = -1;
for (int i = 0; i < this.records.length; i++) {
if (this.records[i].orderNum == orderNum) {
n = i;
}
}
return n;
}
}
class Table {
int tableNum;
Order order;
Date date;
double dicount = 0;
}
public static boolean timeCalendar(Date nowTime, Date beginTime, Date endTime) {
Calendar date = Calendar.getInstance();// 设置当前时间
date.setTime(nowTime);
Calendar begin = Calendar.getInstance();// 设置开始时间
begin.setTime(beginTime);// 开始时间
Calendar end = Calendar.getInstance();// 设置结束时间
end.setTime(endTime);// 上午结束时间
if ((date.after(begin) && date.before(end))) {// 判断是否处于开始时间之后和结束时间之前
return true;
} else {
return false;
}
}
}
代码分析:
-
创建菜品类:Dish
- 属性:菜品名称(name)、基础价格(unit_price)
- 方法:getPrice(int portion) 用于计算菜品价格
-
创建菜谱类:Menu
- 属性:菜品数组dishs,保存所有菜品信息
- 方法:searchDish(String dishName) 根据菜名在菜谱中查找菜品信息,返回Dish对象;addDish(String dishName, int unit_price) 添加一道菜品信息
-
创建点菜记录类:Record
- 属性:序号(orderNum)、菜品(d)、份额(portion)
- 方法:getPrice() 计价,计算本条记录的价格
-
创建订单类:Order
- 属性:记录数组records,保存订单上每一道的记录
- 方法:getTotalPrice() 计算订单的总价;addARecord(int orderNum, String dishName, int portion, int num) 添加一条菜品信息到订单中;delARecordByOrderNum(int orderNum) 根据序号删除一条记录;findRecordByNum(int orderNum) 根据序号查找一条记录
-
设计主程序类:MainClass
- 处理输入信息,包括菜单、订单等
- 根据输入的信息,计算每一桌的总价
- 考虑异常情况,并给出相应的输出信息
在程序设计中,需要注意以下几点:
- 对于菜谱信息与订单信息混合的情况,需要忽略夹在订单信息中的菜谱信息,并输出"invalid dish"
- 要处理桌号格式的合法性,如年必须是4位数字,月、日、时、分、秒可以是1位或2位数
- 合并同一桌菜名、份额相同的点菜记录
- 处理各种异常情况,包括份额超出范围、份数超出范围、桌号超出范围等
- 异常处理机制:try 结构用于创建一个异常处理块,它包含可能引发异常的代码。当异常发生时,程序会跳转到 try 块之后的 catch 块来处理异常。
- 最终输出每一桌的总价,根据折扣计算并四舍五入得到整数金额
![]()
菜单计价程序-5
-
7-1 菜单计价程序-5分数 100作者 蔡轲单位 南昌航空大学
本题在菜单计价程序-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 formatimport java.util.ArrayList; import java.util.List; import java.util.Scanner; import java.time.LocalDateTime; public class Main { public static void main(String[] args) { Table[] table=new Table[10]; Menu menu = new Menu(); Scanner input = new Scanner(System.in); String nextLine = input.nextLine(); int i=0; int num=0; int flag=0; int temp=0; int sametime=0; while (!nextLine.equals("end")) { String[] lineArray = nextLine.split(" "); if(nextLine.equals("")) { nextLine = input.nextLine(); System.out.println("wrong format"); continue; } else if(lineArray.length == 7&&lineArray[0].equals("table")==true&&canParseInt(lineArray[1])==true &&isopen(lineArray[5],lineArray[6])==true&&judge(lineArray[4])){ i++; flag=1; num=0; sametime=0; table[i]=new Table(); table[i].order=new Order(menu); table[i].num=Integer.parseInt(lineArray[1]); table[i].name=lineArray[3]; table[i].phoneNum=lineArray[4]; table[i].time=new Time(); table[i].time.time1=lineArray[5]; table[i].time.time2=lineArray[6]; System.out.println("table "+Integer.parseInt(lineArray[1])+": "); temp=0; } else if (lineArray.length == 7&&lineArray[0].equals("table")==true&&(!judge(lineArray[4])||canParseInt(lineArray[1])==false)) { System.out.println("wrong format"); temp=1; } else if(lineArray.length == 7&&lineArray[0].equals("table")==true&&(canParseInt(lineArray[1])==false||Integer.parseInt(lineArray[1])>55||Integer.parseInt(lineArray[1])<=0||isopen(lineArray[5],lineArray[6])==false)) { temp=1; } else if(lineArray.length >7&&lineArray[0].equals("table")==true) { System.out.println("wrong format"); temp=1; } else if ((lineArray.length == 4||lineArray.length == 5)&&lineArray[0].equals("table")==false&&temp==0&&canParseInt(lineArray[0])) { int orderNum = Integer.parseInt(lineArray[0]); String dishName = lineArray[1]; int parseInt =0; int parseInt1 =0; int parseInt2 =0; if(lineArray.length == 4){ parseInt1 = Integer.parseInt(lineArray[2]); parseInt2 = Integer.parseInt(lineArray[3]); }else { parseInt = Integer.parseInt(lineArray[2]); parseInt1 = Integer.parseInt(lineArray[3]); parseInt2 = Integer.parseInt(lineArray[4]); } // if(lineArray[0].length()>1&&Integer.parseInt(lineArray[0])<10) // System.out.println("wrong format"); // else { if(sametime==0&&table[i].order.addARecord(orderNum, dishName, parseInt, parseInt1,parseInt2,i,false,false)!=null) num=orderNum; else if(sametime==1){ num=0; table[i-1].order.addARecord(orderNum, dishName, parseInt, parseInt1,parseInt2,i-1,false,false); table[i].sametime=1; } // } } else if ("delete".equals(lineArray[1])&&temp==0) { table[i].order.delARecordByOrderNum(Integer.parseInt(lineArray[0]),i); } else if((lineArray.length ==5||lineArray.length ==6)&&canParseInt(lineArray[0])==true&&canParseInt(lineArray[1])==true){ int a=0; if(i>1){ for(int j=1;j<=i;j++){ if(table[j].num==Integer.parseInt(lineArray[0])){ Dish dish = menu.searthDish(lineArray[2]); int price=0; if(lineArray.length ==5) price= dish.getPrice(Integer.parseInt(lineArray[3]))*Integer.parseInt(lineArray[4]); else price= dish.getPrice(Integer.parseInt(lineArray[4]))*Integer.parseInt(lineArray[5]); System.out.println(lineArray[1]+" table "+table[i].num+" pay for table "+table[j].num+" "+price); table[Integer.parseInt(lineArray[0])].order.addARecord(Integer.parseInt(lineArray[1]),lineArray[2],Integer.parseInt(lineArray[3]),Integer.parseInt(lineArray[4]),Integer.parseInt(lineArray[5]),Integer.parseInt(lineArray[0]),true,false); table[i].order.addARecord(Integer.parseInt(lineArray[1]),lineArray[2],Integer.parseInt(lineArray[3]),Integer.parseInt(lineArray[4]),Integer.parseInt(lineArray[5]),i,false,true); a=1; } } if(a==0) System.out.println("Table number :"+Integer.parseInt(lineArray[0])+" does not exist"); } else System.out.println("Table number :"+Integer.parseInt(lineArray[0])+" does not exist"); } else { if((lineArray.length == 3)&&canParseInt(lineArray[0])==false&&lineArray[1].equals("delete")==false) { System.out.println("wrong format"); } if(lineArray.length == 4&&canParseInt(lineArray[2])==true&&lineArray[3].equals("T")) menu.addDish(lineArray[0], lineArray[1],Integer.parseInt(lineArray[2]),true); if(lineArray.length == 2&&canParseInt(lineArray[1])==true&&flag==0) menu.addDish(lineArray[0], null,Integer.parseInt(lineArray[1]),false); } if(lineArray.length == 7&&lineArray[0].equals("table")==true&&canParseInt(lineArray[1])==true&&isopen(lineArray[5], lineArray[6]) == false) { if (isopen(lineArray[5], lineArray[6]) == false ) System.out.println("table " + Integer.parseInt(lineArray[1]) + " out of opening hours"); } nextLine = input.nextLine(); } input.close(); for(int j=1;j<=i;j++){ table[j].getprice(j); } for(int j=1;j<=i;j++){ for(int k=j+1;k<=i;k++) { if (table[j].name!=null&&table[k].name!=null&&table[j].name.compareTo(table[k].name) == 0){ table[k].name=null; table[j].Tableprice+=table[k].Tableprice; } if(table[j].name!=null&&table[k].name!=null&&table[j].name.compareTo(table[k].name)>0){ table[9]=table[j]; table[j]=table[k]; table[k]=table[9]; } } } for(int j=1;j<=i;j++){ if(table[j].name!=null) System.out.println(table[j].name+" "+table[j].phoneNum+" "+table[j].Tableprice); } } public static boolean canParseInt(String str) { if(str==null) { return false; } return str.matches("\\d+"); } public static boolean judge(String str){ if(str.length()!=11)return false; String[] strs=str.split(""); if(strs[0].equals("1")&&strs[1].equals("8")&&strs[2].equals("0"))return true; if(strs[0].equals("1")&&strs[1].equals("8")&&strs[2].equals("1"))return true; if(strs[0].equals("1")&&strs[1].equals("8")&&strs[2].equals("9"))return true; if(strs[0].equals("1")&&strs[1].equals("3")&&strs[2].equals("3"))return true; if(strs[0].equals("1")&&strs[1].equals("3")&&strs[2].equals("5"))return true; if(strs[0].equals("1")&&strs[1].equals("3")&&strs[2].equals("6"))return true; return false; } public static boolean isopen(String str ,String str2){ Time time = new Time(); time.time1=str; time.time2=str2; time.getDay(); time.getYear(); time.getweekOfDay(); if (time.weekday<=5&&time.weekday>=1&&((time. hour>=17&&time.hour<20)||(time. hour==20&&time .minute<=30)||(time.hour==10&&time.minute>=30)||(time.hour>=11&&time.hour<14)||(time.hour==14&&time.minute<=30))) { return true; } else if((time. weekday==6|| time . weekday==7)&&((time.hour==9&&time . minute>=30)|| (time.hour>9&&time.hour<21)||(time. hour==21&&time . minute<=30))) { return true; }else { return false; } } } class Menu { private List<Dish> dishs = new ArrayList<>();//菜品数组,保存所有菜品信息 Dish searthDish(String dishName) { for (Dish dish : dishs) { if (dish.getDishname().equals(dishName)) { return dish; } } return null; } //添加一道菜品信息 Dish addDish(String dishName,String kindOfDish, int unit_price,boolean t) { for (Dish dish : dishs) { if (dish.getDishname().equals(dishName)) { dish.setUnit_price(unit_price); dish.setKindOfDish(kindOfDish); dish.setT(t); return dish; } } Dish dish = new Dish(dishName,kindOfDish, unit_price,t); dishs.add(dish); return dish; } } class Dish { String dishname;//菜品名称 int unit_price; //单价 String kindOfDish; int tastenum; public void setKindOfDish(String kindOfDish) { this.kindOfDish = kindOfDish; } boolean T; public void setT(boolean t) { T = t; } public String getDishname() { return dishname; } public void setUnit_price(int unit_price) { this.unit_price = unit_price; } public Dish(String name, String kindOfDish,int unit_price,boolean t) { this.dishname = name; this.kindOfDish=kindOfDish; this.unit_price = unit_price; this.T=t; } //计算菜品价格的方法,输入参数是点菜的份额(输入数据只能是1/2/3,代表小/中/大份) int getPrice(int portion) { if (portion == 2) return (int) Math.round(1.5 *unit_price); else if (portion == 3) return 2 * unit_price ; else return unit_price ; } } class Record { private int table; private int numOrder;//序号\ private Dish d;//菜品\ private int portion;//份额(1/2/3代表小/中/大份)\ public int num; public int degree; public boolean isreplace; public boolean bereplace; private boolean isDelete = false; private int deleteNum=0; public int getDeleteNum() { return deleteNum; } public void setDeleteNum(int deleteNum) { this.deleteNum = deleteNum; } public boolean isNotFound() { return notFound; } private boolean notFound = false; public Record(int orderNum, Dish d,int degree, int portion, int num) { this.numOrder = orderNum; this.d = d; d.tastenum=degree; this.degree = degree; this.portion = portion; this.num = num; } //计价,计算本条记录的价格 int getPrice() { return d.getPrice(portion) * this.num; } public int getNumOrder() { return numOrder; } public Dish getD() { return d; } public void setDelete(boolean delete) { isDelete = delete; } public boolean isDelete() { return isDelete; } } class Order { private Menu menu; public boolean iscc; public boolean isjc; public boolean iszc; public int ccNum=0; public int jcNum=0; public int zcNum=0; public static Record[][] getRecords() { return records; } public int ccDegree=0; public int jcDegree=0; public int zcDegree=0; static Record[][] records=new Record[10][40]; public Order(Menu menu) { this.menu = menu; } //计算订单的总价 int getTotalPrice(int i) { int sum = 0; for (int j=1;j<=records[i].length;j++) { if(records[i][j]==null)break; int price = records[i][j].getPrice(); if (!records[i][j].isDelete()&&records[i][j].getD().T==false&&records[i][j].isreplace==false) { sum = sum + price; } } return sum; } int getTotalPrice2(int i) { int sum = 0; for (int j=1;j<=records[i].length;j++) { if(records[i][j]==null)break; int price = records[i][j].getPrice(); if (!records[i][j].isDelete()&&records[i][j].getD().T==true&&records[i][j].isreplace==false) { sum = sum + price; } } return sum; } int getTotalPrice3(int i,double a) { int sum = 0; for (int j=1;j<=records[i].length;j++) { if(records[i][j]==null)break; int price = (int)Math.round(records[i][j].getPrice()*a); if (!records[i][j].isDelete()&&records[i][j].getD().T==false&&records[i][j].isreplace==false) { sum = sum + price; } } return sum; } int getTotalPrice4(int i,double a) { int sum = 0; for (int j=1;j<=records[i].length;j++) { if(records[i][j]==null)break; int price = (int)Math.round(records[i][j].getPrice()*a); if (!records[i][j].isDelete()&&records[i][j].getD().T==true&&records[i][j].isreplace==false) { sum = sum + price; } } return sum; } //添加一条菜品信息到订单中。 Record addARecord(int orderNum, String dishName, int degree,int portion, int num,int i,boolean isreplace,boolean bereplace) { Dish dish = menu.searthDish(dishName); if (dish == null) { System.out.println(dishName + " does not exist"); return null; } if(/*(dish.isT()==true&&portion==2)||*/portion>3) { System.out.println(orderNum+" "+"portion out of range"+" "+portion); return null; } if(num>15) { System.out.println(orderNum+" "+"num out of range"+" "+num); return null; } if(dish.kindOfDish!=null&&dish.kindOfDish.equals("川菜")&&(degree>5||degree<0)) { System.out.println("spicy num out of range :" + degree); return null; } if(dish.kindOfDish!=null&&dish.kindOfDish.equals("晋菜")&&(degree>4||degree<0)) { System.out.println("acidity num out of range :" + degree); return null; } if(dish.kindOfDish!=null&&dish.kindOfDish.equals("浙菜")&&(degree>3||degree<0)) { System.out.println("sweetness num out of range :" + degree); return null; } if(dish.kindOfDish!=null&&dish.kindOfDish.equals("川菜")&&(degree<=5&°ree>=0)&&bereplace==false) { this.iscc=true; this.ccDegree+=degree*num; this.ccNum+=num; } if(dish.kindOfDish!=null&&dish.kindOfDish.equals("晋菜")&&(degree<=4&°ree>=0)&&bereplace==false) { this.isjc=true; this.jcDegree+=degree*num; this.jcNum+=num; } if(dish.kindOfDish!=null&&dish.kindOfDish.equals("浙菜")&&(degree<=3&°ree>=0)&&bereplace==false) { this.iszc=true; this.zcDegree+=degree*num; this.zcNum+=num; } int t = 0; for (int j=1;j<=records[i].length;j++) { if(records[i][j]==null) { t=j; break; } } records[i][t]= new Record(orderNum, dish,degree, portion, num); int price = records[i][t].getPrice(); records[i][t].isreplace=isreplace; if(records[i][t].isreplace==false&&bereplace==false) System.out.println(records[i][t].getNumOrder() + " " + records[i][t].getD().getDishname() + " " + price); return records[i][t]; } public boolean delARecordByOrderNum(int orderNum,int i) { int t=0; for (int j=1;j<=20;j++) { if (records[i][j]!=null&&!records[i][j].isNotFound() &&records[i][j].getNumOrder() == orderNum) { records[i][j].setDelete(true); records[i][j].setDeleteNum(records[i][j].getDeleteNum()+1); if(records[i][j].getD().kindOfDish!=null&&records[i][j].getD().kindOfDish.equals("川菜")&&this.iscc) { this.ccDegree-=records[i][j].degree*records[i][j].num; this.ccNum-=records[i][j].num; if(this.ccDegree==0&&this.ccNum==0) this.iscc=false; } if(records[i][j].getD().kindOfDish!=null&&records[i][j].getD().kindOfDish.equals("晋菜")&&this.isjc) { this.jcDegree-=records[i][j].degree*records[i][j].num; this.jcNum-=records[i][j].num; if(this.jcDegree==0&&this.jcNum==0) this.isjc=false; } if(records[i][j].getD().kindOfDish!=null&&records[i][j].getD().kindOfDish.equals("浙菜")&&this.iszc) { this.zcDegree-=records[i][j].degree*records[i][j].num; this.zcNum-=records[i][j].num; if(this.zcDegree==0&&this.zcNum==0) this.iszc=false; } t=records[i][j].getDeleteNum(); if(t>1) { System.out.println("deduplication "+orderNum); } return true; } } System.out.println("delete error;"); return false; } } class Table{ int num; Time time ; Order order; long Tableprice; String name; String phoneNum; int sametime=0; int price=0; void getprice(int i) { time.getDay(); time.getYear(); time.getweekOfDay(); if (time.weekday<=5&&time.weekday>=1) { if((time. hour>=17&&time.hour<20)||(time. hour==20&&time .minute<=30) ) { this.Tableprice=this.order.getTotalPrice3(i,0.8)+this.order.getTotalPrice4(i,0.7); System.out.print("table "+this.num+": "+(this.order.getTotalPrice(i)+this.order.getTotalPrice2(i))+" "+(this.Tableprice+price)); } else if((time.hour==10&&time.minute>=30)||(time.hour>=11&&time.hour<14)||(time.hour==14&&time.minute<=30)) {this.Tableprice=this.order.getTotalPrice3(i,0.6)+this.order.getTotalPrice4(i,0.7); System. out. print("table "+this. num+": "+(this.order.getTotalPrice(i)+this.order.getTotalPrice2(i))+" "+(this.Tableprice+price)) ; } } if(time. weekday==6|| time . weekday==7) { if((time.hour==9&&time . minute>=30)|| (time.hour>9&&time.hour<21)||(time. hour==21&&time . minute<=30) ) { Tableprice=Math. round ((order.getTotalPrice(i)+order.getTotalPrice2(i))); System. out. print ("table "+this. num+": "+(this.order.getTotalPrice(i)+this.order.getTotalPrice2(i))+" "+(this.Tableprice+price)) ; } } if(this.order.iscc) { System.out.print(" 川菜 " + this.order.ccNum ); int a=(int)Math.round(this.order.ccDegree/(this.order.ccNum*1.0) ); if(a==0) System.out.print(" 不辣"); if(a==1) System.out.print(" 微辣"); if(a==2) System.out.print(" 稍辣"); if(a==3) System.out.print(" 辣"); if(a==4) System.out.print(" 很辣"); if(a==5) System.out.print(" 爆辣"); } if(this.order.isjc) { System.out.print(" 晋菜 " + this.order.jcNum ); int a=(int)Math.round(this.order.jcDegree/(this.order.jcNum*1.0) ); if(a==0) System.out.print(" 不酸"); if(a==1) System.out.print(" 微酸"); if(a==2) System.out.print(" 稍酸"); if(a==3) System.out.print(" 酸"); if(a==4) System.out.print(" 很酸"); } if(this.order.iszc) { System.out.print(" 浙菜 " + this.order.zcNum ); int a=(int)Math.round(this.order.zcDegree/(this.order.zcNum*1.0) ); if(a==0) System.out.print(" 不甜"); if(a==1) System.out.print(" 微甜"); if(a==2) System.out.print(" 稍甜"); if(a==3) System.out.print(" 甜"); } if(!this.order.iszc&&!this.order.isjc&&!this.order.iscc) System.out.print(" "); System.out.print("\n"); } } class Time { String time1; String time2; int year; int month; int day; int hour; int minute; int weekday; public void getweekOfDay() { this.weekday=LocalDateTime.of(this.year, this.month, this.day, this.hour, this.minute).getDayOfWeek().getValue(); } public void getYear() { String Date1[] = time1.split("\\/"); year = Integer.parseInt(Date1[0]); month = Integer.parseInt(Date1[1]); day = Integer.parseInt(Date1[2]); } public void getDay() { String Date2[] = time2.split("\\/"); hour = Integer.parseInt(Date2[0]); minute = Integer.parseInt(Date2[1]); } }代码分析:
-
Menu类:这个类表示了菜单,包含了菜品和价格的信息。在这个类中,你定义了一个字典来存储菜品和对应的价格。这样可以方便地查找菜品和其价格。 -
Order类:这个类表示了订单,包含了订单号、菜品、数量等信息。其中有一个方法用来计算订单的总价。这个类的设计使得每个订单都可以独立管理自己的信息。 -
Table类:这个类表示了餐桌,包含了餐桌号和订单信息。通过这个类,可以将订单和餐桌进行关联。1. 用户交互 -
主程序通过命令行或者其他适当的用户界面与用户进行交互,接收用户输入并根据输入执行相应的操作。用户可以进行创建订单、添加菜品、删除菜品、计算价格等操作。
-
主程序通过循环来持续接收用户的输入,并根据输入来调用相应的方法或函数。在接收到用户输入后,主程序会解析用户的指令,并根据指令来执行相应的操作,比如创建订单、添加菜品、打印订单详情等。
-
主程序在执行用户指令时,会调用 Menu 类、Order 类和 Table 类等相关对象的方法,来实现具体的功能。比如创建订单时会调用 Table 类的
create_order方法,添加菜品时会调用 Table 类的add_item_to_order方法等。 -
主程序需要对用户输入进行合法性检查,并进行相应的错误处理。当用户输入不合法或操作失败时,主程序需要给出相应的提示信息,并引导用户进行正确的操作。
-
为了提高系统的可扩展性,主程序应该尽可能地遵循单一职责原则,保持简洁清晰的逻辑结构,并提供良好的接口,以便将来能够方便地扩展新的功能或修改现有功能。
![]()
![]()
期中测试:7-4测验4-抽象类与接口
在测验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.62import java.util.Scanner; public class Main { public static void main(String[] args) { Scanner input = new Scanner(System.in); Shape[] shapes = new Shape[100]; // 假设最多输入100个图形 int count = 0; int choice = input.nextInt(); while (choice != 0 && count < 100) { // 防止数组越界 switch (choice) { case 1: double radius = input.nextDouble(); shapes[count] = new Circle(radius); count++; 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); shapes[count] = new Rectangle(leftTopPoint, lowerRightPoint); count++; break; } choice = input.nextInt(); } for (int i = 0; i < count; i++) { for (int j = i + 1; j < count; j++) { if (shapes[i].calculateArea() > shapes[j].calculateArea()) { Shape temp = shapes[i]; shapes[i] = shapes[j]; shapes[j] = temp; } } } for (int i = 0; i < count; i++) { System.out.print(String.format("%.2f", shapes[i].calculateArea()) + " "); } } } interface Shape { double calculateArea(); } class Circle implements Shape { private double radius; public Circle(double radius) { this.radius = radius; } public double calculateArea() { return Math.PI * radius * radius; } } 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 implements Shape { private Point leftTop; private Point lowerRight; public Rectangle(Point leftTop, Point lowerRight) { this.leftTop = leftTop; this.lowerRight = lowerRight; } public double calculateArea() { double length = Math.abs(lowerRight.getX() - leftTop.getX()); double width = Math.abs(lowerRight.getY() - leftTop.getY()); return length * width; } }分析:
这段代码是一个简单的图形面积计算程序,通过用户输入创建圆形或矩形对象,并计算它们的面积。让我们来分析一下这段代码的主要功能和结构:
1. 主程序 Main
main 方法接收用户的输入,创建 Circle 或 Rectangle 对象,并计算它们的面积。
通过 while 循环持续接收用户输入,直到用户输入 0 或者已经输入了 100 个图形。
2. 图形接口 Shape
定义了一个 calculateArea 方法,用于计算图形的面积。
3. 圆形类 Circle
实现了 Shape 接口,包含了一个 radius 属性和 calculateArea 方法来计算圆形的面积。
4. 点类 Point
简单的表示二维平面上的一个点,包含 x 和 y 坐标属性。
5. 矩形类 Rectangle
同样实现了 Shape 接口,包含两个点对象 leftTop 和 lowerRight,通过这两个点来计算矩形的面积。
6. 功能分析
用户可以输入 1 来创建圆形,输入半径信息;输入 2 来创建矩形,输入左上角和右下角坐标信息。
创建的图形对象被存储在 shapes 数组中,然后按照面积从小到大的顺序进行排序,并输出它们的面积。![]()
![]()
改进建议:
菜单计价:
为了使系统更加完善,可以考虑以下改进与扩展:
- 数据持久化:引入文件存储或数据库,实现订单信息的持久化。
- 异常处理:在用户交互和操作过程中加入异常处理机制。
- 菜品库存管理:扩展菜单类,加入菜品库存管理的功能。
- 用户界面:实现一个友好的图形界面或网页界面。
- 订单状态管理:引入订单状态管理的功能,例如已下单、已支付、已上菜等状态。
- 单元测试:为关键部分的代码编写单元测试,确保代码的质量和稳定性。
图形面积计算:
- 在数组排序部分使用的是简单的冒泡排序算法,对于大量数据会有性能问题,可以考虑更高效的排序算法。
- 输入过程没有做异常处理,如用户输入非数字字符会导致程序异常
总结:
经过这一阶段的学习,对于JAVA编程算是有了一个比较大的进步,但是写菜单计价程序这种难度很高的编程题还是感觉像刚开始一样有一种无从下手的感觉,好在对于期中考试中的编程题还是能够去全部完成。对于这一阶段的学习我发现了学习JAVA 需要做到的一些细节点掌握基础概念:在开始学习Java之前,确保你对基本的编程概念(如变量、循环、条件语句等)有一定的了解,这将为你学习Java打下良好的基础。实践与项目:学习编程最重要的一点是不断地实践。尝试编写小程序,深入理解面向对象编程:Java是一种面向对象的语言,因此深入理解面向对象编程思想对于学习Java至关重要。包括类、对象等概念。
-






浙公网安备 33010602011771号