前言:
菜单计价程序-3作为计价4和计价5的基础,做不了3就不用谈作为延伸拓展的4和5,在第一次做菜单计价3时,因为时间安排的原因,菜单计价3连及格的分都没有拿到,还是在补练中写完拿的满分,菜单计价程序的这几个还是有一定难度的,我大概每个菜单的题目都用了两三个晚上写了有五百行左右写完。本次由于3写的不是很好,导致后面4,5想优化都要几乎重写,花费了不少时间。期中考试难度一般。主要是菜单4涉及到了异常的处理机制
设计与分析:
7-1 菜单计价程序-3 分数 40 作者 蔡轲 单位 南昌航空大学设计点菜计价程序,根据输入的信息,计算并输出总价格。
输入内容按先后顺序包括两部分:菜单、订单,最后以"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
代码长度限制
16 KB
时间限制
400 ms
内存限制
64 MB
import java.time.LocalDate; import java.time.LocalTime; import java.time.format.DateTimeFormatter; import java.time.format.DateTimeFormatterBuilder; import java.util.Scanner; class BusinessHours { static final LocalTime OPENING_TIME_WEEKDAYS = LocalTime.of(9, 29,59); static final LocalTime CLOSING_TIME_WEEKDAYS = LocalTime.of(21, 31); static final LocalTime DISCOUNT_START_TIME_EVENING = LocalTime.of(16, 59,59); static final LocalTime DISCOUNT_END_TIME_EVENING = LocalTime.of(20, 31); static final LocalTime DISCOUNT_START_TIME_LUNCH = LocalTime.of(10, 29,59); static final LocalTime DISCOUNT_END_TIME_LUNCH = LocalTime.of(14, 31); static boolean isOpen(LocalDate date, LocalTime time) { // 判断是否为周末 if (date.getDayOfWeek().getValue() >= 6) { return time.isAfter(OPENING_TIME_WEEKDAYS) && time.isBefore(CLOSING_TIME_WEEKDAYS); } // 判断晚上折扣时间段 if (time.isAfter(DISCOUNT_START_TIME_EVENING) && time.isBefore(DISCOUNT_END_TIME_EVENING)) { return true; } // 判断中午折扣时间段 if (time.isAfter(DISCOUNT_START_TIME_LUNCH) && time.isBefore(DISCOUNT_END_TIME_LUNCH)) { return true; } return false; } public static double getDiscount(LocalDate date, LocalTime time) { // 周一至周五晚上8折 if (date.getDayOfWeek().getValue() < 6 && time.isAfter(DISCOUNT_START_TIME_EVENING) && time.isBefore(DISCOUNT_END_TIME_EVENING)) { return 0.8; } // 周一至周五中午6折 if (date.getDayOfWeek().getValue() < 6 && time.isAfter(DISCOUNT_START_TIME_LUNCH) && time.isBefore(DISCOUNT_END_TIME_LUNCH)) { return 0.6; } // 其他时间无折扣 return 1.0; } } class Dish { String name; int unit_price; public Dish(String name,int unit_price){ this.name = name; this.unit_price = unit_price; } public int getPrice(int portion){ return Math.round(unit_price * (((portion - 1) * 0.5f) + 1) ); } } class Menu { public Dish[] dishes = new Dish[20]; int countdish=0; void addDish(String dishName, int unit_price){ dishes[countdish++] = new Dish(dishName,unit_price); } Dish searchDish(String dishName) { Dish temp = null; for (int i = countdish-1 ; i >=0 ;i-- ) { if (dishes[i].name.equals(dishName)) { temp = dishes[i]; break; } } if(temp == null) System.out.println(dishName+" does not exist"); return temp; } } class Record { int orderNum, portion, getPrice, num; Dish d; public Record(int orderNum, int portion, int num, Dish d) { this.orderNum = orderNum; this.portion = portion; this.num = num; this.d = d; } } class Order { Record[] records=new Record[20]; int count = 0; int allprice = 0; public Record addaRecord(Menu menu, int orderNum, String dishName, int portion, int num){ Record record = new Record(orderNum, portion, num, menu.searchDish(dishName) ); if(record.d != null) { record.getPrice = record.d.getPrice(portion) * num; System.out.println(orderNum +" "+ dishName + " " + record.getPrice); records[count++] = record; allprice += record.getPrice; }else records[count++] = record; return record; } public Record AddOrtherRecord(Menu menu, int orderNum, String dishName, int portion, int num,int tableid,int nowtable){ Record record = new Record(orderNum, portion, num, menu.searchDish(dishName) ); if(record.d != null) { record.getPrice = record.d.getPrice(portion) * num; // System.out.println(orderNum +" "+ dishName + " " + record.getPrice); System.out.println(orderNum + " table " + nowtable + " pay for table "+ tableid + " " + record.getPrice); records[count++] = record; allprice += record.getPrice; } return record; } public void delARecordByOrderNum(int orderNum){ if(orderNum > count || orderNum < 1) System.out.println("delete error;"); else allprice -= records[orderNum-1].getPrice; } } class Table { int id; Order order; LocalDate date; LocalTime time; int price; double discount = 1; public Table(int id, Order order, LocalDate date, LocalTime time) { this.id = id; this.order = order; this.date = date; this.time = time; } void Gettottalprice() { if (BusinessHours.isOpen(date, time)) { discount = BusinessHours.getDiscount(date, time); if (discount > 0) { price = (int)Math.round(order.allprice * discount); System.out.println("table " + id + ": " + price); } } else { System.out.println("table " + id + " out of opening hours"); } } } class Main{ public static void main(String[] args){ int JudgeInputLenth; Scanner scanner=new Scanner(System.in); Table[] table = new Table[20]; int TableCount = 0; Menu menu = new Menu(); DateTimeFormatter formatter = new DateTimeFormatterBuilder() .appendOptional(DateTimeFormatter.ofPattern("yyyy/MM/dd")) .appendOptional(DateTimeFormatter.ofPattern("yyyy/M/dd")) .appendOptional(DateTimeFormatter.ofPattern("yyyy/MM/d")) .appendOptional(DateTimeFormatter.ofPattern("yyyy/M/d")) .toFormatter(); DateTimeFormatter timeFormatter = new DateTimeFormatterBuilder() .appendOptional(DateTimeFormatter.ofPattern("HH/mm/ss")) .appendOptional(DateTimeFormatter.ofPattern("HH/m/ss")) .appendOptional(DateTimeFormatter.ofPattern("HH/mm/s")) .appendOptional(DateTimeFormatter.ofPattern("HH/m/s")) .appendOptional(DateTimeFormatter.ofPattern("H/m/s")) .toFormatter(); while (true) { String line = scanner.nextLine().trim(); if (line.equals("end")) break; String[] parts = line.split(" "); JudgeInputLenth = parts.length; if (parts[0].equals("table")) {//创建桌号 int tableNum = Integer.parseInt(parts[1]); String dateStr = parts[2]; String timeStr = parts[3]; LocalDate date = LocalDate.parse(dateStr, formatter); LocalTime time = LocalTime.parse(timeStr, timeFormatter); table[TableCount++] = new Table(tableNum, new Order(), date, time); System.out.println("table " + tableNum + ": "); } else if (Character.isDigit(parts[0].charAt(0)) && JudgeInputLenth == 2) {//第一个为数字且长度为2 //删除 table[TableCount - 1].order.delARecordByOrderNum(Integer.parseInt(parts[0])); } else if (Character.isDigit(parts[0].charAt(0)) && JudgeInputLenth == 4) {//第一个为数字且长度为4 //点餐 int orderNum = Integer.parseInt(parts[0]); String dishName = parts[1]; int portion = Integer.parseInt(parts[2]); int num = Integer.parseInt(parts[3]); table[TableCount - 1].order.addaRecord(menu, orderNum, dishName, portion, num); }else if(JudgeInputLenth == 5){ int tableid = Integer.parseInt(parts[0]); int orderNum = Integer.parseInt(parts[1]); String dishName = parts[2]; int portion = Integer.parseInt(parts[3]); int num = Integer.parseInt(parts[4]); table[TableCount - 1].order.AddOrtherRecord(menu, orderNum, dishName, portion, num,tableid,TableCount); //4 table 2 pay for table 1 12 //1 4 麻婆豆腐 1 1 } else {//菜单加菜 String dishName = parts[0]; int unit_price = Integer.parseInt(parts[1]); menu.addDish(dishName, unit_price); } } for (int i = 0; i < TableCount; i++) { table[i].Gettottalprice(); } } }
踩坑心得:本次建议还是多使用一些java提供的类,不然极其容易运行超时
在本次使用了BusinessHours
用于检查企业在特定日期和时间是否营业以及根据提供的日期和时间计算折扣百分比的方法
LocalDate
表示没有时区的日期。LocalTime
表示没有时区的时间。
DateTimeFormatter
用于格式化和解析日期和时间。
DateTimeFormatterBuilder
用于构建自定义的DateTimeFormatter
实例。
isOpen(LocalDate date, LocalTime time)
方法:检查在指定的日期和时间业务是否营业。
首先检查该天是否是周末(星期六或星期天)。如果是,它会检查时间是否在工作日的常规营业时间内(从OPENING_TIME_WEEKDAYS
到CLOSING_TIME_WEEKDAYS
)。
如果是工作日,它会检查时间是否在晚间折扣时段(从DISCOUNT_START_TIME_EVENING
到DISCOUNT_END_TIME_EVENING
)或午餐折扣时段(从DISCOUNT_START_TIME_LUNCH
到DISCOUNT_END_TIME_LUNCH
)
该方法如果企业营业则返回true
,否则返回false
。
getDiscount(LocalDate date, LocalTime time)
方法:此方法基于提供的日期和时间计算并返回折扣百分比。
如果是工作日晚上,折扣设置为0.8。如果是工作日午餐时间,折扣设置为0.6。
如果不在指定的折扣时段内,则折扣设置无折扣。
DateTimeFormatter formatter = new DateTimeFormatterBuilder()时间格式化器
DateTimeFormatter timeFormatter = new DateTimeFormatterBuilder()日期格式化器
由于输入的年月日的格式并不固定,所以添加了可能的格式用于接收解析
其他的类的设计基本上还是使用了菜单计价2的思路
最后也是在补练中解决了在当时大作业一直存在的运行超时问题。
改进建议:主函数的Table可以用arraylist代替普通数组,可以减少增删改查的操作
7-1 菜单计价程序-4 分数 100 作者 蔡轲 单位 南昌航空大学
本体大部分内容与菜单计价程序-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 MBimport java.time.DateTimeException; import java.time.LocalDate; import java.time.LocalTime; import java.time.format.DateTimeFormatter; import java.time.format.DateTimeFormatterBuilder; import java.util.Scanner; class BusinessHours { static final LocalTime OPENING_TIME_WEEKDAYS = LocalTime.of(9, 29,59); static final LocalTime CLOSING_TIME_WEEKDAYS = LocalTime.of(21, 31); static final LocalTime DISCOUNT_START_TIME_EVENING = LocalTime.of(16, 59,59); static final LocalTime DISCOUNT_END_TIME_EVENING = LocalTime.of(20, 31); static final LocalTime DISCOUNT_START_TIME_LUNCH = LocalTime.of(10, 29,59); static final LocalTime DISCOUNT_END_TIME_LUNCH = LocalTime.of(14, 31); static boolean isOpen(LocalDate date, LocalTime time) { // 判断是否为周末 if (date.getDayOfWeek().getValue() >= 6) { return time.isAfter(OPENING_TIME_WEEKDAYS) && time.isBefore(CLOSING_TIME_WEEKDAYS); } // 判断晚上折扣时间段 if (time.isAfter(DISCOUNT_START_TIME_EVENING) && time.isBefore(DISCOUNT_END_TIME_EVENING)) { return true; } // 判断中午折扣时间段 if (time.isAfter(DISCOUNT_START_TIME_LUNCH) && time.isBefore(DISCOUNT_END_TIME_LUNCH)) { return true; } return false; } public static double getDiscount(LocalDate date, LocalTime time) { // 周一至周五晚上8折 if (date.getDayOfWeek().getValue() < 6 && time.isAfter(DISCOUNT_START_TIME_EVENING) && time.isBefore(DISCOUNT_END_TIME_EVENING)) { return 0.8; } // 周一至周五中午6折 if (date.getDayOfWeek().getValue() < 6 && time.isAfter(DISCOUNT_START_TIME_LUNCH) && time.isBefore(DISCOUNT_END_TIME_LUNCH)) { return 0.6; } // 其他时间无折扣 return 1.0; } public static double getTDiscount(LocalDate date, LocalTime time) { // 周一至周五晚上7折 if (date.getDayOfWeek().getValue() < 6) { return 0.7; } return 1.0; } } class Dish { String name; int unit_price; String TE; public Dish(String name, int unit_price, String TE) { this.name = name; this.unit_price = unit_price; this.TE = TE; } public int getPrice(int portion){ return Math.round(unit_price * (((portion - 1) * 0.5f) + 1) ); } } class Menu { public Dish[] dishes = new Dish[20]; int countdish=0; void addDish(String dishName, int unit_price,String Te){ dishes[countdish++] = new Dish(dishName,unit_price,Te); } Dish searchDish(String dishName) { Dish temp = null; for (int i = countdish-1 ; i >=0 ;i-- ) { if (dishes[i].name.equals(dishName)) { temp = dishes[i]; break; } } return temp; } } class Record { int orderNum, portion, getPrice, num; Dish d; double discount; public Record(double discount) { this.discount = discount; } public Record(int orderNum, int portion, int num, Dish d) { this.orderNum = orderNum; this.portion = portion; this.num = num; this.d = d; } } class Order { Record[] records=new Record[20]; int count = 0;//订单数 int[] trueRecordcount = new int[20];//真正有效的订单数de编号 int allprice = 0;//总价 int[] delnum = new int[20]; int delnumcount = 0; int discountAllPrice = 0;//打折之后的总价 LocalDate date; LocalTime time; public Order(LocalDate date, LocalTime time) { this.date = date; this.time = time; } public Record addaRecord(Menu menu, int orderNum, Dish dish, int portion, int num){ Record record = new Record(orderNum, portion, num, dish ); records[count++] = record; if(record.d != null) { record.getPrice = record.d.getPrice(portion) * num; if(record.d.TE != null) { DiscountPrice(record.getPrice, true); } else{ DiscountPrice(record.getPrice, false); } System.out.println(orderNum +" "+ dish.name + " " + record.getPrice); trueRecordcount[count] = orderNum; allprice += record.getPrice; } return record; } public void DiscountPrice(int price,boolean isTESE){ double discount; if(!isTESE) { discount = BusinessHours.getDiscount(date, time); } else { discount = BusinessHours.getTDiscount(date, time); } records[count - 1].discount = discount; if (discount > 0) { price = (int) Math.round(price * discount); discountAllPrice += price; } } public Record AddOrtherRecord(Menu menu, int orderNum, String dishName, int portion, int num,int tableid,int nowtable,LocalDate date, LocalTime time){ Record record = new Record(orderNum, portion, num, menu.searchDish(dishName) ); this.date = date; this.time = time; if(record.d != null) { record.getPrice = record.d.getPrice(portion) * num; if(record.d.TE != null) { DiscountPrice(record.getPrice, true); } else{ DiscountPrice(record.getPrice, false); } System.out.println(orderNum + " table " + nowtable + " pay for table "+ tableid + " " + record.getPrice); records[count++] = record; allprice += record.getPrice; } return record; } public void SearchRecord(Dish dish, int portion, int num){ Record srecord = null; for (int i = 0 ; i < count ; i++ ) { if (records[i].d.name.equals(dish.name) ) { if(records[i].portion == portion) { srecord = records[i]; break; } } } //records[count++] = record; if(srecord != null) { srecord.num += num; // if(srecord.d.TE != null) { // DiscountPrice(srecord.getPrice, true); // } // else{ // DiscountPrice(srecord.getPrice, false); // } // //System.out.println(orderNum +" "+ dish.name + " " + record.getPrice); // //trueRecordcount++; // allprice += srecord.getPrice; } // return srecord; } public boolean addDelNum(int orderNum){ for (int i = 0; i < delnumcount; i++) { if(orderNum == delnum[i]){ System.out.println("deduplication " + orderNum); return false; } } delnum[delnumcount++] = orderNum; return true; } public void delARecordByOrderNum(int orderNum) { if (addDelNum(orderNum)) { if (orderNum > count || orderNum < 1) System.out.println("delete error;"); else { allprice -= records[orderNum - 1].getPrice; discountAllPrice -= (records[orderNum - 1].getPrice * records[orderNum - 1].discount); } } } } class Table { int id; Order order; int price; double discount = 1; public Table(int id, Order order) { this.id = id; this.order = order; } void Gettottalprice() { price = order.allprice; int discountprice = order.discountAllPrice; System.out.println("table " + id + ": " + price + " " + discountprice); if (!BusinessHours.isOpen(order.date, order.time)) { System.out.println("table " + id + " out of opening hours"); } } } public class Main{ public static void main(String[] args){ int JudgeInputLenth; Scanner scanner=new Scanner(System.in); Table[] table = new Table[20]; int TableCount = -1;//已定的桌数 //double discount = 0;//折扣 int orderNum;//点菜编号 String dishName ;//菜名 int portion ;//份额 int num;//数量 int tableid;//桌号 boolean availtable = false; Menu menu = new Menu(); DateTimeFormatter formatter = new DateTimeFormatterBuilder() .appendOptional(DateTimeFormatter.ofPattern("yyyy/MM/dd")) .appendOptional(DateTimeFormatter.ofPattern("yyyy/M/dd")) .appendOptional(DateTimeFormatter.ofPattern("yyyy/MM/d")) .appendOptional(DateTimeFormatter.ofPattern("yyyy/M/d")) .toFormatter(); DateTimeFormatter timeFormatter = new DateTimeFormatterBuilder() .appendOptional(DateTimeFormatter.ofPattern("HH/mm/ss")) .appendOptional(DateTimeFormatter.ofPattern("HH/m/ss")) .appendOptional(DateTimeFormatter.ofPattern("HH/mm/s")) .appendOptional(DateTimeFormatter.ofPattern("HH/m/s")) .appendOptional(DateTimeFormatter.ofPattern("H/m/s")) .toFormatter(); while (true) { String line = scanner.nextLine().trim(); if (line.equals("end")) break; String[] parts = line.split(" "); JudgeInputLenth = parts.length; try { if (JudgeInputLenth == 4 && parts[0].equals("table")) {//创建桌号 TableCount++; int tableNum = Integer.parseInt(parts[1]); if(parts[2].length() > 10 || parts[3].length() > 8){ throw new NumberFormatException(); } String dateStr = parts[2]; String timeStr = parts[3]; try { TableIDException(tableNum);//桌号范围判断 if(tableNum < 1 ){ throw new Exception(); } LocalDate date = LocalDate.parse(dateStr, formatter); LocalTime time = LocalTime.parse(timeStr, timeFormatter); validtimeException(date);//日期是否有效 ZeroFirstWordException(parts[1],parts[0]);//最高位是否为零 String[] judge1 = parts[2].split("/"); String[] judge2 = date.toString().split("-"); for (int i = 0; i < judge1.length; i++) { if(Integer.parseInt(judge1[i]) != Integer.parseInt(judge2[i])){ throw new DateTimeException(parts[1]); } } if (BusinessHours.isOpen(date, time)) { availtable = true; table[TableCount] = new Table(tableNum, new Order(date, time)); System.out.println("table " + tableNum + ": "); } else { System.out.println("table " + tableNum + " out of opening hours"); } } catch (ZeroFirstException zfe) { System.out.println("wrong format"); }catch (DateTimeException dte) {//时间格式 System.out.println(tableNum + " date error"); }catch (TableOutOfRangeException toore){ System.out.println(tableNum +" table num out of range"); }catch (TimeOutOfRangeException toore){ System.out.println("not a valid time period"); }catch (Exception e) { System.out.println("wrong format"); } } else if ((JudgeInputLenth < 2) || (JudgeInputLenth > 5) || (!parts[0].equals("table")&& parts[0].startsWith("t"))){//cuowu菜单加菜 System.out.println("wrong format"); } else if (((JudgeInputLenth == 4) && !Character.isDigit(parts[0].charAt(0))) &&(!Character.isDigit(parts[2].charAt(0)) || !parts[0].equals("table") || parts[2].length() > 10 || parts[3].length() > 8 ||parts[2].length() < 8|| parts[3].length() <5) ){//cuowu菜单加菜 System.out.println("wrong format"); availtable = false; }else if ((JudgeInputLenth >= 5) && parts[0].equals("table")){ System.out.println("wrong format"); availtable = false; } else if( parts[0].equals("") ||(JudgeInputLenth ==2 &&(parts[0].equals("") || parts[1].equals("") )&&TableCount == -1) || (JudgeInputLenth ==3 && (parts[0].equals("") || parts[1].equals("") ||parts[2].equals("") )&& TableCount == -1)|| (JudgeInputLenth ==4 && (parts[0].equals("") || parts[1].equals("") ||parts[2].equals("") ||parts[3].equals(""))&& TableCount == -1)|| (JudgeInputLenth ==5 && (parts[0].equals("") || parts[1].equals("") ||parts[2].equals("") ||parts[3].equals("")||parts[4].equals("") )&& TableCount == -1)){ System.out.println("wrong format"); availtable = false; } else if( parts[0].equals("") || (JudgeInputLenth ==2 && TableCount != -1 && table[TableCount] != null &&(parts[0].equals("") || parts[1].equals("") )) || (JudgeInputLenth ==3 && TableCount != -1 && table[TableCount] != null && (parts[0].equals("") || parts[1].equals("") ||parts[2].equals("") ))|| (JudgeInputLenth ==4 && TableCount != -1 && table[TableCount] != null && (parts[0].equals("") || parts[1].equals("") ||parts[2].equals("") ||parts[3].equals("")) )|| (JudgeInputLenth ==5 && TableCount != -1 && table[TableCount] != null && (parts[0].equals("") || parts[1].equals("") ||parts[2].equals("") ||parts[3].equals("")||parts[4].equals("") ))){ System.out.println("wrong format"); availtable = false; } else if (JudgeInputLenth == 2 && Character.isDigit(parts[0].charAt(0)) && (TableCount != -1 && table[TableCount] != null)) {//第一个为数字且长度为2 //删除 table[TableCount].order.delARecordByOrderNum(Integer.parseInt(parts[0])); } else if (availtable && JudgeInputLenth == 4 && Character.isDigit(parts[0].charAt(0)) &&(TableCount != -1 && table[TableCount] != null) ) {//第一个为数字且长度为4 //点餐 dishName = parts[1]; portion = Integer.parseInt(parts[2]); orderNum = Integer.parseInt(parts[0]); num = Integer.parseInt(parts[3]); try { isDishException(menu,dishName);//菜名错误 portionException(portion);//份额超出范围 numException(num); //份数超出范围 orderNumException(table[TableCount].order,orderNum); //序号不是从小到大 ZeroFirstWordException(parts[3],parts[0]);//最高位是否为零 Dish dish = menu.searchDish(dishName); table[TableCount].order.addaRecord(menu, orderNum, dish, portion, num); }catch (ZeroFirstException zfe) { System.out.println("wrong format"); } catch (OrderNumException one) { System.out.println( "record serial number sequence error" ); } catch (PortionOutOfRangeException poore) { table[TableCount].order.count++; System.out.println(orderNum + " portion out of range " + portion); } catch (NumOutOfRangeException noore){ table[TableCount].order.count++; System.out.println(orderNum +" num out of range "+ num); } catch (NotFoundDishException nfde){ table[TableCount].order.count++; System.out.println(dishName + " does not exist"); } } else if (JudgeInputLenth == 5 && table[TableCount] != null && Character.isDigit(parts[0].charAt(0))) {//代点餐 tableid = Integer.parseInt(parts[0]); orderNum = Integer.parseInt(parts[1]); dishName = parts[2]; portion = Integer.parseInt(parts[3]); num = Integer.parseInt(parts[4]); boolean ViladorderNum = false; for (int i = 0; i < TableCount; i++) { if(tableid == table[i].id){ ViladorderNum = true; break; } } if(ViladorderNum == true) { LocalDate date = table[TableCount - 1].order.date; LocalTime time = table[TableCount - 1].order.time; table[TableCount].order.AddOrtherRecord(menu, orderNum, dishName, portion, num, tableid, TableCount, date, time); }else System.out.println("Table number :" + tableid + " does not exist"); } else if ((JudgeInputLenth == 2 || JudgeInputLenth == 3) && parts[1] != null && Character.isDigit(parts[1].charAt(0)) && TableCount != -1&& table[TableCount] != null) { System.out.println("invalid dish"); } else if ((JudgeInputLenth == 2 || (JudgeInputLenth == 3 && parts[2] != null &&parts[2].equals("T")))&& parts[0] != null && parts[1] != null &&!Character.isDigit(parts[0].charAt(0)) && Character.isDigit(parts[1].charAt(0))&& TableCount == -1) {//菜单加菜 dishName = parts[0]; String tese = null; int unit_price = Integer.parseInt(parts[1]); if (JudgeInputLenth == 3) { tese = parts[2]; } try{ ZeroFirstWordException(parts[1],parts[0]); unit_priceOutrangeException(unit_price); menu.addDish(dishName, unit_price, tese); }catch (PriceOutrangeException poe) { System.out.println(dishName + " price out of range " + unit_price); } catch (ZeroFirstException zfe) { System.out.println("wrong format"); } } else if ((JudgeInputLenth == 2 || (JudgeInputLenth == 3 && !parts[2].equals("T")))&& !Character.isDigit(parts[0].charAt(0))&& TableCount == -1) {//cuowu菜单加菜 System.out.println("wrong format"); } } catch(NumberFormatException e) { System.out.println("wrong format"); } } for (int i = 0; i <= TableCount; i++) { if(table[i] != null) { table[i].Gettottalprice(); } } } public static void portionException(int portion)throws PortionOutOfRangeException { if(portion>3||portion<1){ throw new PortionOutOfRangeException(); } } public static void numException(int num)throws NumOutOfRangeException { if(num > 15|| num < 1 ){ throw new NumOutOfRangeException(); } } public static void TableIDException(int num)throws TableOutOfRangeException { if(num > 55 ){ throw new TableOutOfRangeException(); } } public static void validtimeException( LocalDate date)throws TimeOutOfRangeException { LocalDate start = LocalDate.parse("2022-01-01"); LocalDate end = LocalDate.parse("2023-12-31"); if(!(date.isAfter(start) && date.isBefore(end))){ throw new TimeOutOfRangeException(); } } public static void isDishException(Menu menu, String name)throws NotFoundDishException { if(menu.searchDish(name) == null){ throw new NotFoundDishException(); } } public static void orderNumException(Order order, int ordernum)throws OrderNumException { if(order.trueRecordcount[order.count] != 0 && order.trueRecordcount[order.count] >= ordernum){ // if(order.trueRecordcount != 0 && order.records[order.count - 1].orderNum >= ordernum){ throw new OrderNumException(); } } public static void ZeroFirstWordException(String one, String two)throws ZeroFirstException { if(one.charAt(0) == '0'|| two.charAt(0) == '0'){ throw new ZeroFirstException(); } } public static void unit_priceOutrangeException(int price)throws PriceOutrangeException { if(price < 1 || price > 300){ throw new PriceOutrangeException(); } } public static class PriceOutrangeException extends Exception{} public static class ZeroFirstException extends Exception{} public static class OrderNumException extends Exception{} public static class NotFoundDishException extends Exception{} public static class TimeOutOfRangeException extends Exception{} public static class TableOutOfRangeException extends Exception{} public static class PortionOutOfRangeException extends Exception{} public static class NumOutOfRangeException extends Exception{} }
本次相比于菜单3添加了特色菜,并统一了计价的方式,并添加了许多异常情况,由图可以看到本次的平均复杂度还是比以前好的
Java使用 throw 关键字抛出一个 异常类(Exception) 或者 异常类的子类 的实例,表示异常的发生,
需要使用try{ }catch(){ } 语句:
1、将可能出现异常的操作放在 try{} 部分;
2、一旦 try{} 部分抛出异常对象 或 调用某个可能抛出异常对象的方法,那么 try{} 部分将立刻停止执行,转向执行相应catch部分;
3、程序将发生异常后的处理放在 catch(){} 部分;
下面截取了本题中我用到的一个异常,大概流程如下
try { ...... numException(num); //份数超出范围
...... }catch (NumOutOfRangeException noore){ table[TableCount].order.count++; System.out.println(orderNum +" num out of range "+ num); } ......
public static void numException(int num)throws NumOutOfRangeException { if(num > 15|| num < 1 ){ throw new NumOutOfRangeException(); } }
public static class NumOutOfRangeException extends Exception{}
本次的难点对我来说是如何准确的知道输入的语句是什么命令,因为一些多(少)打了字等异常情况,所以用Scanner接收的字符数组的长度判断已经不够准确了,在本次作业中我是用 if 语句里面疯狂添加&&以及 | | 来缩小范围让语句正确的执行。我的判断条件确实是写了一大堆,后面想改也难,
可能的改进:现在想了一下或许直接用正则表达式就可以解决。
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 format
代码长度限制
50 KB
时间限制
1000 ms
内存限制
64 MB
import java.time.LocalDate; import java.time.LocalTime; import java.time.format.DateTimeFormatter; import java.time.format.DateTimeFormatterBuilder; import java.util.Scanner; import java.util.*; class BusinessHours { static final LocalTime OPENING_TIME_WEEKDAYS = LocalTime.of(9, 29,59); static final LocalTime CLOSING_TIME_WEEKDAYS = LocalTime.of(21, 31); static final LocalTime DISCOUNT_START_TIME_EVENING = LocalTime.of(16, 59,59); static final LocalTime DISCOUNT_END_TIME_EVENING = LocalTime.of(20, 31); static final LocalTime DISCOUNT_START_TIME_LUNCH = LocalTime.of(10, 29,59); static final LocalTime DISCOUNT_END_TIME_LUNCH = LocalTime.of(14, 31); static boolean isOpen(LocalDate date, LocalTime time) { // 判断是否为周末 if (date.getDayOfWeek().getValue() >= 6) { return time.isAfter(OPENING_TIME_WEEKDAYS) && time.isBefore(CLOSING_TIME_WEEKDAYS); } // 判断晚上折扣时间段 if (time.isAfter(DISCOUNT_START_TIME_EVENING) && time.isBefore(DISCOUNT_END_TIME_EVENING)) { return true; } // 判断中午折扣时间段 if (time.isAfter(DISCOUNT_START_TIME_LUNCH) && time.isBefore(DISCOUNT_END_TIME_LUNCH)) { return true; } return false; } public static double getDiscount(LocalDate date, LocalTime time) { // 周一至周五晚上8折 if (date.getDayOfWeek().getValue() < 6 && time.isAfter(DISCOUNT_START_TIME_EVENING) && time.isBefore(DISCOUNT_END_TIME_EVENING)) { return 0.8; } // 周一至周五中午6折 if (date.getDayOfWeek().getValue() < 6 && time.isAfter(DISCOUNT_START_TIME_LUNCH) && time.isBefore(DISCOUNT_END_TIME_LUNCH)) { return 0.6; } // 其他时间无折扣 return 1.0; } public static double getTDiscount(LocalDate date) { // 周一至周五晚上7折 if (date.getDayOfWeek().getValue() < 6) { return 0.7; } return 1.0; } } class Dish { String name; int unit_price; String T; String type; public Dish(String name, int unit_price, String t, String type) { this.name = name; this.unit_price = unit_price; T = t; this.type = type; } public int getPrice(int portion){ return Math.round(unit_price * (((portion - 1) * 0.5f) + 1) ); } } class Menu { public Dish[] dishes = new Dish[20]; int countdish=0; void addDish(String dishName, int unit_price,String T, String type){ dishes[countdish++] = new Dish(dishName,unit_price,T,type); } Dish searchDish(String dishName) { Dish temp = null; for (int i = countdish-1 ; i >=0 ;i-- ) { if (dishes[i].name.equals(dishName)) { temp = dishes[i]; break; } } if(temp == null) System.out.println(dishName+" does not exist"); return temp; } } class Record { int orderNum, portion, getPrice, num,degree; Dish d; double discount; public boolean JudgeDegree(){ int min = 0; int max = 5;//默认为川菜的辣度 if (d.type.equals("晋菜")) { max = 4; if(degree > max || degree < min){ System.out.println("acidity num out of range :"+ degree); return false; } } else if (d.type.equals("浙菜")) { max = 3; if(degree > max || degree < min) { System.out.println("sweetness num out of range :" + degree); //sweetness return false; } } if(degree > max || degree < min){ System.out.println("spicy num out of range :" + degree);//spicy/ return false; } return true; } public Record(int orderNum, int portion, int num, Dish d,int degree) { this.orderNum = orderNum; this.portion = portion; this.num = num; this.d = d; this.degree = degree; } } class Taste{ int AllSpicy = 0;//总辣度 int SpicyCount = 0;//川菜份数 int AvgSpicy = 0;//平均辣度 int Spicynum = 0; int AllAcidity = 0;//总酸度 int AcidityCount = 0;//晋菜份数 int AvgAcidity = 0;//平均酸度 int Aciditynum = 0; int AllSweetness = 0;//总甜度 int SweetnessCount = 0;//浙菜份数 int AvgSweetness = 0;//平均甜度 int Sweetnessnum = 0; String chuan_taste; String jin_taste; String zhe_taste; //川菜增加辣度值:辣度0-5级;对应辣度水平为:不辣、微辣、稍辣、辣、很辣、爆辣; // //晋菜增加酸度值,酸度0-4级;对应酸度水平为:不酸、微酸、稍酸、酸、很酸; // //浙菜增加甜度值,甜度0-3级;对应酸度水平为:不甜、微甜、稍甜、甜; public void addtaste(String name, int degree,int num){ if(name.equals("川菜")){ AllSpicy += num * degree; SpicyCount++; Spicynum += num; } else if (name.equals("晋菜")) { AllAcidity += num * degree; AcidityCount++; Aciditynum += num; } else if (name.equals("浙菜")) { AllSweetness += num * degree; SweetnessCount++; Sweetnessnum += num; } } public void deltaste(String name, int degree,int num){ if(name.equals("川菜")){ AllSpicy -= num * degree; SpicyCount--; Spicynum -= num; } else if (name.equals("晋菜")) { AllAcidity -= num * degree; AcidityCount--; Aciditynum -= num; } else if (name.equals("浙菜")) { AllSweetness -= num * degree; SweetnessCount--; Sweetnessnum -= num; } } public void tasteset(){ if(SpicyCount!=0) { AvgSpicy = (int)Math.round(AllSpicy * 1.0 / Spicynum); switch (AvgSpicy) { case 0: chuan_taste = "不辣"; break; case 1: chuan_taste = "微辣"; break; case 2: chuan_taste = "稍辣"; break; case 3: chuan_taste = "辣"; break; case 4: chuan_taste = "很辣"; break; case 5: chuan_taste = "爆辣"; break; } }if(AcidityCount != 0) { AvgAcidity = (int )Math.round(AllAcidity * 1.0 / Aciditynum); switch (AvgAcidity) {//不酸、微酸、稍酸、酸、很酸 case 0: jin_taste = "不酸"; break; case 1: jin_taste = "微酸"; break; case 2: jin_taste = "稍酸"; break; case 3: jin_taste = "酸"; break; case 4: jin_taste = "很酸"; break; } }if(SweetnessCount != 0) { AvgSweetness = (int )Math.round(AllSweetness * 1.0 / Sweetnessnum); switch (AvgSweetness) { case 0: zhe_taste = "不甜"; break; case 1: zhe_taste = "微甜"; break; case 2: zhe_taste = "稍甜"; break; case 3: zhe_taste = "甜"; break; } } } } class Order { Record[] records=new Record[20]; int count = 0; int allprice = 0; LocalDate date;//订单日期 LocalTime time;//订单时间 int discountAllPrice = 0;//打折之后的总价 int[] trueRecordcount = new int[20];//真正有效的订单数de编号 int[] delnum = new int[20]; int delnumcount = 0; public Order(LocalDate date, LocalTime time) { this.date = date; this.time = time; } public Record addaRecord(int lenth,Taste taste, int orderNum, Dish dish, int portion, int num,int degree){//添加菜品 Record record = new Record(orderNum, portion, num,dish ,degree); boolean IsVaildDegree = true; if( lenth == 5 ) { IsVaildDegree = record.JudgeDegree(); if(IsVaildDegree) taste.addtaste(record.d.type, degree,num); } records[count++] = record; if( IsVaildDegree) { record.getPrice = record.d.getPrice(portion) * num; if(record.d.T != null) { DiscountPrice(record.getPrice, true); } else{ DiscountPrice(record.getPrice, false); } System.out.println(orderNum +" "+ record.d.name + " " + record.getPrice); trueRecordcount[count] = orderNum; allprice += record.getPrice; } return record; } public Record AddOrtherRecord(Table table ,int lenth,Menu menu, int orderNum, String dishName, int portion, int num,int tableid,int nowtable,int degree){ Record record = new Record(orderNum, portion, num, menu.searchDish(dishName) ,degree); boolean IsVaildDegree = true; double discount; if(record.d != null && lenth == 6 ) { IsVaildDegree = record.JudgeDegree(); if(IsVaildDegree) table.taste.addtaste(record.d.type, degree,num); } if(record.d != null && IsVaildDegree) { record.getPrice = record.d.getPrice(portion) * num; if(record.d.T != null) { discount = BusinessHours.getTDiscount(date); } else{ discount = BusinessHours.getDiscount(date, time); } //records[count - 1].discount = discount; if (discount > 0) { int price = (int) Math.round(record.getPrice * discount); discountAllPrice += price; } System.out.println(orderNum + " table " + nowtable + " pay for table "+ tableid + " " + record.getPrice); records[count++] = record; allprice += record.getPrice; } return record; } public void DiscountPrice(int price,boolean isTESE){//计算打折之后的总价 double discount; if(!isTESE) { discount = BusinessHours.getDiscount(date, time); } else { discount = BusinessHours.getTDiscount(date); } records[count - 1].discount = discount; if (discount > 0) { price = (int) Math.round(price * discount); discountAllPrice += price; } } public boolean addDelNum(int orderNum){ for (int i = 0; i < delnumcount; i++) { if(orderNum == delnum[i]){ //System.out.println("deduplication " + orderNum); System.out.println("delete error;"); return false; } } delnum[delnumcount++] = orderNum; return true; } public void delARecordByOrderNum(Table table,int orderNum) { boolean a = addDelNum(orderNum); if (a) { boolean isviladnum = false; int temp = 0; for (int i = table.order.count - 1; i >= 0; i--) { if(table.order.records[i] != null && table.order.records[i].orderNum == orderNum){ isviladnum = true; temp = i; break; } } if (!isviladnum) System.out.println("delete error;"); else { // if(record.d != null && lenth == 5 ) { // IsVaildDegree = record.JudgeDegree(); // if(IsVaildDegree) // taste.addtaste(record.d.type, degree,num); // } int num = table.order.records[temp].num; if(table.order.records[temp].degree != 0 && table.order.records[temp].degree < 6) { int degree = table.order.records[temp].degree; String type = table.order.records[temp].d.type; table.taste.deltaste(type, degree, num); } allprice -= records[orderNum - 1].getPrice; //discountAllPrice -= (int) (records[orderNum - 1].getPrice * records[orderNum - 1].discount);//43 // discountAllPrice = (int) (allprice * records[orderNum - 1].discount);//39\ discountAllPrice = 0; records[temp].getPrice = 0; for (int i = 0; i < count; i++) { discountAllPrice += (int) Math.round(records[i].getPrice * records[i].discount); } } } } } class Table { int id;//桌号 Order order;//订单 int price;//价格 String Name; String PhoneNumber; Taste taste ; int discountprice; public Table(int id, Order order, String name, String phoneNumber,Taste taste) { this.id = id; this.order = order; Name = name; PhoneNumber = phoneNumber; this.taste = taste; } void Gettottalprice() { price = order.allprice; discountprice = order.discountAllPrice; taste.tasteset(); //table+英文空格+桌号+“:”+英文空格+当前桌的计算折扣后总价+英文空格+辣度平均值+英文空格+酸度平均值+英文空格+甜度平均值+英文空格 //table 1: 113 113 川菜 1 稍辣 晋菜 4 稍酸 浙菜 1 甜 //table 2: 168 118 川菜 4 稍辣 晋菜 2 微酸 //table 3: 103 73 川菜 1 爆辣 晋菜 1 稍酸 浙菜 3 微甜 if (taste.Spicynum != 0 && taste.Aciditynum != 0 && taste.Sweetnessnum != 0) {System.out.println("table " + id + ": " + price + " " + discountprice + " " + "川菜" + " " + taste.Spicynum + " " + taste.chuan_taste + " " + "晋菜" + " " + taste.Aciditynum + " " + taste.jin_taste + " " + "浙菜" + " " + taste.Sweetnessnum + " " + taste.zhe_taste); } else if (taste.Spicynum == 0 && taste.Aciditynum == 0 && taste.Sweetnessnum == 0) {System.out.println("table " + id + ": " + price + " " + discountprice + " "); } else if (taste.Spicynum != 0 && taste.Aciditynum != 0 && taste.Sweetnessnum == 0) {System.out.println("table " + id + ": " + price + " " + discountprice + " " + "川菜" + " " + taste.Spicynum + " " + taste.chuan_taste + " " + "晋菜" + " " + taste.Aciditynum + " " + taste.jin_taste); } else if (taste.Spicynum != 0 && taste.Aciditynum == 0 && taste.Sweetnessnum != 0) {System.out.println("table " + id + ": " + price + " " + discountprice + " " + "川菜" + " " + taste.Spicynum + " " + taste.chuan_taste + " " + "浙菜" + " " + taste.Sweetnessnum + " " + taste.zhe_taste); } else if (taste.Spicynum == 0 && taste.Aciditynum != 0 && taste.Sweetnessnum != 0) {System.out.println("table " + id + ": " + price + " " + discountprice + " " + "晋菜" + " " + taste.Aciditynum + " " + taste.jin_taste + " " + "浙菜" + " " + taste.Sweetnessnum + " " + taste.zhe_taste); } else if (taste.Spicynum != 0 && taste.Aciditynum == 0 && taste.Sweetnessnum == 0) {System.out.println("table " + id + ": " + price + " " + discountprice + " " + "川菜" + " " + taste.Spicynum + " " + taste.chuan_taste); } else if (taste.Spicynum == 0 && taste.Aciditynum != 0 && taste.Sweetnessnum == 0) {System.out.println("table " + id + ": " + price + " " + discountprice + " " + "晋菜" + " " + taste.Aciditynum + " " + taste.jin_taste); } else if (taste.Spicynum == 0 && taste.Aciditynum == 0 && taste.Sweetnessnum != 0) {System.out.println("table " + id + ": " + price + " " + discountprice + " " + "浙菜" + " " + taste.Sweetnessnum + " " + taste.zhe_taste); } } } class Main{ public static void main(String[] args){ int JudgeInputLenth; Scanner scanner=new Scanner(System.in); Table[] table = new Table[20]; int TableCount = 0;//桌数记录 int orderNum = 0; //点菜编号 String dishName = null; //菜名 int portion = 0; //份额 int num = 0; //份数 int unit_price = 0;//菜品基础价格 String name; //订餐人姓名 String phonenum ; //电话号码 boolean availtable = false; Menu menu = new Menu(); DateTimeFormatter formatter = new DateTimeFormatterBuilder() .appendOptional(DateTimeFormatter.ofPattern("yyyy/MM/dd")) .appendOptional(DateTimeFormatter.ofPattern("yyyy/M/dd")) .appendOptional(DateTimeFormatter.ofPattern("yyyy/MM/d")) .appendOptional(DateTimeFormatter.ofPattern("yyyy/M/d")) .toFormatter(); DateTimeFormatter timeFormatter = new DateTimeFormatterBuilder() .appendOptional(DateTimeFormatter.ofPattern("HH/mm/ss")) .appendOptional(DateTimeFormatter.ofPattern("HH/m/ss")) .appendOptional(DateTimeFormatter.ofPattern("HH/mm/s")) .appendOptional(DateTimeFormatter.ofPattern("HH/m/s")) .appendOptional(DateTimeFormatter.ofPattern("H/m/s")) .toFormatter(); while (true) { String line = scanner.nextLine().trim(); if (line.equals("end")) break; String[] parts = line.split(" "); JudgeInputLenth = parts.length; if (((JudgeInputLenth < 2) || (JudgeInputLenth > 7)) && parts[0].equals("table")){//cuowu System.out.println("wrong format"); }else if (parts[0].equals("table")) {//创建桌号 table 1 : tom 13670008181 2023/5/1 21/30/00 int tableNum = Integer.parseInt(parts[1]); name = parts[3]; phonenum = parts[4]; String dateStr = parts[5]; String timeStr = parts[6]; LocalDate date = LocalDate.parse(dateStr, formatter); LocalTime time = LocalTime.parse(timeStr, timeFormatter); //约束条件:客户姓名不超过10个字符,手机号11位,前三位必须是180、181、189、133、135、136其中之一。 if(name.length() < 11 && phonenum.length() < 12 &&(phonenum.matches("^(180|181|189|133|135|136)\\d{8}$"))){ if (BusinessHours.isOpen(date, time)) { availtable = true; table[TableCount++] = new Table(tableNum, new Order( date, time), name,phonenum,new Taste()); System.out.println("table " + tableNum + ": "); } else { System.out.println("table " + tableNum + " out of opening hours"); } }else System.out.println("wrong format"); } else if (availtable &&Character.isDigit(parts[0].charAt(0)) && JudgeInputLenth == 2) {//第一个为数字且长度为2 //删除 table[TableCount - 1].order.delARecordByOrderNum(table[TableCount - 1],Integer.parseInt(parts[0])); } else if (availtable && Character.isDigit(parts[0].charAt(0)) && !Character.isDigit(parts[1].charAt(0))) {//第一个为数字且长度为4 //点餐 int degree = 0; //口味度 orderNum = Integer.parseInt(parts[0]); dishName = parts[1]; Dish dish = menu.searchDish(dishName); if(JudgeInputLenth == 4 && dish.type == null){ portion = Integer.parseInt(parts[2]); num = Integer.parseInt(parts[3]); } else if (JudgeInputLenth == 5) { degree = Integer.parseInt(parts[2]); portion = Integer.parseInt(parts[3]); num = Integer.parseInt(parts[4]); }else { System.out.println("wrong format"); continue; } if(dish != null) { table[TableCount - 1].order.addaRecord(JudgeInputLenth, table[TableCount - 1].taste, orderNum, dish, portion, num, degree); } }else if(availtable && Character.isDigit(parts[0].charAt(0)) && Character.isDigit(parts[1].charAt(0)) ){//第一和第二个字符都是数字 //代点菜 int degree = 0; //口味度 int tableid = Integer.parseInt(parts[0]); orderNum = Integer.parseInt(parts[1]); dishName = parts[2]; degree = Integer.parseInt(parts[3]); portion = Integer.parseInt(parts[4]); num = Integer.parseInt(parts[5]); int temp = 0;//被带点餐的桌号 for (int i = 0; i < TableCount; i++) { if(table[i].id == tableid){ temp = i; break; } } table[TableCount - 1].order.AddOrtherRecord(table[temp],JudgeInputLenth,menu, orderNum, dishName, portion, num,tableid,TableCount,degree); //1 1 醋浇羊肉 0 1 2 } else if(TableCount == 0 && !availtable){//菜单加菜 String T = null; String type = null; dishName = parts[0]; if(JudgeInputLenth == 2){//东坡肉 25 unit_price = Integer.parseInt(parts[1]); menu.addDish(dishName, unit_price,T,type); } else if (JudgeInputLenth == 3 ||(JudgeInputLenth == 4 && parts[3].equals("2"))) {//东坡肉 25 T System.out.println("wrong format"); } else if (JudgeInputLenth == 4) {//东坡肉 浙菜 25 T type = parts[1]; unit_price = Integer.parseInt(parts[2]); T = parts[3]; menu.addDish(dishName, unit_price,T,type); } } } for (int i = 0; i < TableCount; i++) { table[i].Gettottalprice(); } custommer(table); } public static void custommer(Table[] table){ Map<String, Table> map = new HashMap<>(); for (Table t : table) { if (t != null && t.Name != null) { if (map.containsKey(t.Name)) { // 如果已经存在具有相同名称的表格,合并 discountprice map.get(t.Name).discountprice += t.discountprice; } else { // 否则,添加新表格 map.put(t.Name, t); } } } // 使用 TreeMap 对键进行排序 Map<String, Table> sortedMap = new TreeMap<>(map); for (Map.Entry<String, Table> entry : sortedMap.entrySet()) { Table t = entry.getValue(); System.out.println(t.Name + " " + t.PhoneNumber + " " + t.discountprice); } } }
菜单计价5,感觉上还是不难的,
从图也可以看到本次的图除了最大复杂度没有越过绿色部分的
添加了特色菜的菜系
在3的基础上的dish添加菜系和口味度
在record生成订单时加上菜系和口味度
然后创建一个新的类taste用于计算本桌的口味度
table类加上名字和联系方式
相同的人合并价格以及名字排序这里使用了hashmap,如下
创建了一个HashMap,用于存储表格的名称和对应的Table对象。HashMap的键是表格的名称,值是对应的Table对象。
Map<String, Table> map = new HashMap<>();
循环遍历Table数组:
for (Table t : table) {
if (t != null && t.Name != null) {
if (map.containsKey(t.Name)) {
map.get(t.Name).discountprice += t.discountprice;
} else {
map.put(t.Name, t);
}
}
}
对传入的Table数组进行遍历。如果Table对象不为null且其Name字段不为null,就进行处理。
如果HashMap中已经包含了相同名称的表格,则将当前表格的discountprice累加到已有表格的discountprice上,
否则,将当前表格添加到HashMap中。
使用TreeMap进行排序:TreeMap会根据键的自然顺序或者通过Comparator进行排序。
Map<String, Table> sortedMap = new TreeMap<>(map);
最后遍历并输出结果:
for (Map.Entry<String, Table> entry : sortedMap.entrySet()) {
Table t = entry.getValue();
System.out.println(t.Name + " " + t.PhoneNumber + " " + t.discountprice);
}
使用entrySet遍历TreeMap,按照键的顺序获取Table对象,并输出表格的名称、PhoneNumber和discountprice。
改进:存的Table数组可以用arraylist可以节约一些操作
7-1 测验1-圆类设计 分数 10 作者 段喜龙 单位 南昌航空大学
创建一个圆形类(Circle),私有属性为圆的半径,从控制台输入圆的半径,输出圆的面积
输入格式:
输入圆的半径,取值范围为(0,+∞)
,输入数据非法,则程序输出Wrong Format
,注意:只考虑从控制台输入数值的情况
输出格式:
输出圆的面积(保留两位小数,可以使用String.format(“%.2f”,输出数值)控制精度)
输入样例:
在这里给出一组输入。例如:
2.35
输出样例:
在这里给出相应的输出。例如:
17.35
代码长度限制
16 KB
时间限制
400 ms
内存限制
64 MB
import java.util.Scanner; import static java.lang.System.exit; class Main { public static void main(String[] args) { Scanner in = new Scanner(System.in); double num = in.nextDouble(); Circle Op = new Circle(); Op.setBanjing(num); Op.count(); } } class Circle{ private double banjing; double area; public Circle() { } public double getBanjing() { return banjing; } public void setBanjing(double banjing) { if(banjing > 0 ){ this.banjing = banjing; }else { System.out.println("Wrong Format"); exit(0); } } public void count(){ area = banjing * banjing * 3.1415926535; } }
简单的类的使用,没什么好说的。
7-2 测验2-类结构设计 分数 10 作者 段喜龙 单位 南昌航空大学
设计一个矩形类,其属性由矩形左上角坐标点(x1,y1)及右下角坐标点(x2,y2)组成,其中,坐标点属性包括该坐标点的X轴及Y轴的坐标值(实型数),求得该矩形的面积。类设计如下图:
输入格式:
分别输入两个坐标点的坐标值x1,y1,x2,y2。
输出格式:
输出该矩形的面积值(保留两位小数)。
输入样例:
在这里给出一组输入。例如:
6 5.8 -7 8.9
输出样例:
在这里给出相应的输出。例如:
40.30
代码长度限制
16 KB
时间限制
400 ms
内存限制
64 MB
import java.util.Scanner; import static java.lang.Math.abs; class Main { public static void main(String[] args) { Scanner in = new Scanner(System.in); double num = in.nextDouble(); double num1 = in.nextDouble(); double num2 = in.nextDouble(); double num3 = in.nextDouble(); Point x1 = new Point(); x1.setX(num); x1.setY(num1); Point x2 = new Point(); x2.setX(num2); x2.setY(num3); Rectengle newR = new Rectengle(x1,x2); System.out.println(String.format("%.2f",newR.getarea())); } } class Point{ private double x; private double y; public Point() { } public Point(double x, double y) { this.x = x; this.y = y; } public double getX() { return x; } public void setX(double x) { this.x = x; } public double getY() { return y; } public void setY(double y) { this.y = y; } } class Rectengle{ Point topleftpoint; Point downrightpoint; public Rectengle() { } public Rectengle(Point topleftpoint, Point downrightpoint) { this.topleftpoint = topleftpoint; this.downrightpoint = downrightpoint; } public Point getTopleftpoint() { return topleftpoint; } public void setTopleftpoint(Point topleftpoint) { this.topleftpoint = topleftpoint; } public Point getDownrightpoint() { return downrightpoint; } public void setDownrightpoint(Point downrightpoint) { this.downrightpoint = downrightpoint; } public double getlenth(){ return abs(topleftpoint.getX() - downrightpoint.getX()); } public double getheight(){ return abs(topleftpoint.getY() - downrightpoint.getY()); } public double getarea(){ double lenth = getlenth(); double height = getheight(); return lenth * height; } }
用
abs(topleftpoint.getX() - downrightpoint.getX());
取绝对值
7-3 测验3-继承与多态 分数 20 作者 段喜龙 单位 南昌航空大学
将测验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
代码长度限制
16 KB
时间限制
400 ms
内存限制
64 MB
import java.util.Scanner; import static java.lang.Math.abs; import static java.lang.System.exit; abstract class Shape{ abstract double getArea(); } class Circle extends Shape{ private double radius; public Circle(double radius) { if(radius > 0) this.radius = radius; else{ System.out.println("Wrong Format"); exit(0);} } public double getArea(){ return radius * radius * 3.1415926535; } } 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{ Point topLeftPoint; Point lowerRightPoint; public Rectangle(Point topLeftPoint, Point lowerRightPoint) { this.topLeftPoint = topLeftPoint; this.lowerRightPoint = lowerRightPoint; } public double getLength(){ return abs(topLeftPoint.getX() - lowerRightPoint.getX()); } public double getHeight(){ return abs(topLeftPoint.getY() - lowerRightPoint.getY()); } public double getArea(){ double length = getLength(); double height = getHeight(); return length * height; } } 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(); 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 topLeftPoint = new Point(x1, y1); Point lowerRightPoint = new Point(x2, y2); Rectangle rectangle = new Rectangle(topLeftPoint, lowerRightPoint); printArea(rectangle); break; } } public static void printArea(Shape shape){ System.out.println(String.format("%.2f",shape.getArea())); } }
-
String.format("%.2f", shape.getArea())
: 这将面积值格式化为一个带有两位小数的字符串。%.2f
指定将浮点数格式化为小数点后两位的形式。
public static void printArea(Shape shape){ System.out.println(String.format("%.2f",shape.getArea())); }
7-4 测验4-抽象类与接口 分数 20 作者 段喜龙 单位 南昌航空大学
在测验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
代码长度限制
16 KB
时间限制
400 ms
内存限制
64 MB
import java.util.ArrayList; import java.util.Comparator; import java.util.Scanner; // 定义一个Point类表示坐标点 class Point { double x, y; Point(double x, double y) { this.x = x; this.y = y; } } // 定义一个抽象类Shape,实现Comparable接口 abstract class Shape implements Comparable<Shape> { // 抽象方法,用于获取图形的面积 abstract double getArea(); // 实现compareTo方法,按照面积升序排序 @Override public int compareTo(Shape other) { return Double.compare(this.getArea(), other.getArea()); } } // 定义圆形类 class Circle extends Shape { double radius; Circle(double radius) { this.radius = radius; } @Override double getArea() { return Math.PI * radius * radius; } } // 定义矩形类 class Rectangle extends Shape { Point leftTopPoint, lowerRightPoint; Rectangle(Point leftTopPoint, Point lowerRightPoint) { this.leftTopPoint = leftTopPoint; this.lowerRightPoint = lowerRightPoint; } @Override double getArea() { double width = Math.abs(lowerRightPoint.x - leftTopPoint.x); double height = Math.abs(lowerRightPoint.y - leftTopPoint.y); return width * height; } } // 主类 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: // Circle double radius = input.nextDouble(); Shape circle = new Circle(radius); 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); Shape 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()) + " "); } } }
在考试的时候对接口的实现是真的不熟悉,改来改去也不行,只拿了几分。
上面是课后再写了一遍
Comparable
接口要求实现类具有 compareTo
方法,该方法用于比较当前对象与另一个对象的大小。
Shape
类,该类是其他图形类的父类,且 Shape
类实现了 Comparable
接口:
import java.util.Comparator;
public abstract class Shape implements Comparable<Shape> {
// 其他类的共享属性和方法
public abstract double getArea(); // 假设所有图形都有一个计算面积的方法
@Override
public int compareTo(Shape other) {
// 比较当前对象和另一个对象的面积
double thisArea = this.getArea();
double otherArea = other.getArea();
// 使用 Double.compare 进行比较
return Double.compare(thisArea, otherArea);
}
}
Shape
类实现了 Comparable
接口,并重写了 compareTo
方法。该方法比较两个图形对象的面积。使用 Double.compare
方法可以确保在比较时处理了 NaN 和无穷大等特殊情况。
然后,具体图形类 Circle
和 Rectangle
应该继承 Shape
类,并实现抽象方法 getArea
。图形类就能够通过面积实现比较了。
在调用 list.sort(Comparator.naturalOrder())
时,将使用 compareTo
方法进行比较,并根据面积进行排序。
public class Circle extends Shape {
private double radius;
// 构造函数和其他方法
@Override
public double getArea() {
return Math.PI * radius * radius;
}
}
public class Rectangle extends Shape {
private Point leftTopPoint;
private Point lowerRightPoint;
// 构造函数和其他方法
@Override
public double getArea() {
double width = lowerRightPoint.getX() - leftTopPoint.getX();
double height = lowerRightPoint.getY() - leftTopPoint.getY();
return width * height;
}
}
(5)总结:
这几周在pta上写题目的时间明显多于前几次,首先是因为这几次的菜单计价程序确实是有难度的,同时自己在网上学习了Java编程语言,掌握了面向对象编程的核心概念,如类、对象、继承和多态。熟悉了Java的基本语法、流程控制和异常处理机制。Java的内容很多,不过只是掌握这些是远远不够的,人需努力。
标签:题目,期中考试,PTA,空格,int,parts,&&,table,public From: https://www.cnblogs.com/MysteriousAlter/p/17838500.html