一、前言
1.知识点:
(1)第一次题目集:共有九道题,主要为计算题,考察的内容为Java1~8章节的内容和类的设计。比如浮点型计算, java.util.Scanner方法的运用,if else条件语句、System.out.println分类输 出,且和或条件语句,除法和求余运算,一维数组,排序,多个类的设计;
(2)第二次题目集:主要有三道题,主要考察字符串,数组和类等内容,具体为串口字符串解析,数字与字母的转换以及String的格式判断以及内容提取。
(3)第三次题目集:也是三道题,但难度较大,从点类到线类再到形类,不仅考察了我们数学知识,更让我们更深入体会到了类与对象的关系,并且在理解题目含义的同时运用类的创建(包含 若干属性和若干方法)以及多个类的创建来完成整个代码的框架与思路。
2.题量与难度:
总体上讲题量适中,难度从简单到复杂,很符合我们Java现阶段学习的应用和实践,从第一次作业到第三次,引导我们用不同的方法实现同一项功能,通过具体的题目要求,逐步引导我们从面向过程的思维过渡到面向对象的思维。并且题目类型也比较的基础,所涉及到的知识点和基础语法也是以前C语言中的,只不过像Sccaner和print,即输入输出,Java有它自己的与C不同的代码书写方式,而且像这种经常会用到的函数Java中都整合起来成了一个个方法,需要用的时候只需要一句语句调用即可,更方便快捷。相对于第一次作业,第二次题目集题目难度则更多体现在复杂上。算法的设计都大同小异,但是第三次作业的题目要求数据量则大了许多,涉及到的条件判断也更多。稍不留神少考虑了一种判断情况就造成结果错误。条件判断多,分类情况多。从第三次题目集开始,我们能感觉到正式要踏入面向对象程序设计的大门,前两题的题目要求很明确,严格要求了我们创建类的格式,即包含若干属性和若干方法,而题目的具体要求则是在主类中通过调用具体类的具体方法来实现,最后一题则是涉及多个类的涉及,一个类仅完成一项功能。题量不算多,对于Java课程学习来讲题目难度也应该不算大,但毕竟作为入门选手,从面向过程程序设计跨越到面向对象程序设计,多多少少都会感觉到陌生和有难度。
二、设计与分析
1.期中考试第三题测验3-继承与多态
将测验1与测验2的类设计进行合并设计,抽象出Shape父类(抽象类),Circle及Rectangle作为子类,类图如下所示:

试编程完成如上类图设计,主方法源码如下(可直接拷贝使用):
public static void main(String[] args) {
// TODO Auto-generated method stub
Scanner input = new Scanner(System.in);
int choice = input.nextInt();
switch(choice) {
case 1://Circle
double radiums = input.nextDouble();
Shape circle = new Circle(radiums);
printArea(circle);
break;
case 2://Rectangle
double x1 = input.nextDouble();
double y1 = input.nextDouble();
double x2 = input.nextDouble();
double y2 = input.nextDouble();
Point leftTopPoint = new Point(x1,y1);
Point lowerRightPoint = new Point(x2,y2);
Rectangle rectangle = new Rectangle(leftTopPoint,lowerRightPoint);
printArea(rectangle);
break;
}
}
其中,printArea(Shape shape)方法为定义在Main类中的静态方法,体现程序设计的多态性。
输入格式:
输入类型选择(1或2,不考虑无效输入)
对应图形的参数(圆或矩形)
输出格式:
图形的面积(保留两位小数)
输入样例1:
1
5.6
输出样例1:
在这里给出相应的输出。例如:
98.52
输入样例2:
2
5.6
-32.5
9.4
-5.6
输出样例2:
在这里给出相应的输出。例如:
102.22
代码
import java.util.Scanner; abstract class Shape { abstract double calculateArea(); } class Circle extends Shape { private double radius; public Circle(double radius) { if (radius <= 0) { System.out.println("Wrong Format"); } else { this.radius = radius; } } @Override double calculateArea() { return Math.PI * Math.pow(radius, 2); } } class Point { double x, y; public Point(double x, double y) { this.x = x; this.y = y; } } class Rectangle extends Shape { private Point leftTopPoint, lowerRightPoint; public Rectangle(Point leftTopPoint, Point lowerRightPoint) { this.leftTopPoint = leftTopPoint; this.lowerRightPoint = lowerRightPoint; } @Override double calculateArea() { 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); int choice = input.nextInt(); switch (choice) { case 1:// Circle double radius = input.nextDouble(); Shape circle = new Circle(radius); if (!(circle instanceof Circle)) { return; } 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; } } public static void printArea(Shape shape) { double area = shape.calculateArea(); System.out.println(String.format("%.2f", area)); } }
解释与心得:
这题只用了一个main函数,并没有创建类,而且考察的知识是奇偶校验,判断是否是长度大于最小结构,判断是否有0,开始找0,先判断最后一位是不是结束1,再判断是否奇校验正确;通过两次判断结果来输出。
2.期中考试第四题测验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.62
代码
import java.util.ArrayList; import java.util.Collections; import java.util.Comparator; import java.util.Scanner; interface Shape extends Comparable<Shape> { double getArea(); } class Circle implements Shape { private double radius; public Circle(double radius) { if (radius <= 0) { System.out.println("Wrong Format"); } else { this.radius = radius; } } @Override public double getArea() { return Math.PI * Math.pow(radius, 2); } @Override public int compareTo(Shape o) { return Double.compare(this.getArea(), o.getArea()); } } class Point { double x, y; public Point(double x, double y) { this.x = x; this.y = y; } } class Rectangle implements Shape { private Point leftTopPoint, lowerRightPoint; public Rectangle(Point leftTopPoint, Point lowerRightPoint) { this.leftTopPoint = leftTopPoint; this.lowerRightPoint = lowerRightPoint; } @Override public double getArea() { double width = Math.abs(lowerRightPoint.x - leftTopPoint.x); double height = Math.abs(lowerRightPoint.y - leftTopPoint.y); return width * height; } @Override public int compareTo(Shape o) { return Double.compare(this.getArea(), o.getArea()); } } 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); if (circle instanceof Circle) { 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(); } Collections.sort(list); for (int i = 0; i < list.size(); i++) { System.out.print(String.format("%.2f", list.get(i).getArea()) + " "); } } }
解释与心得:
创建一个point类,设计传参数,返回计算所得到的值。感觉主要的难点还是判断格式和字符判断的问题,首先将两组坐标进行分割,用split函数用空格分割一次后再用split函数将每组坐标中的x,y坐标用逗号分割开。
3.菜单计价程序2
设计点菜计价程序,根据输入的信息,计算并输出总价格。
输入内容按先后顺序包括两部分:菜单、订单,最后以"end"结束。
菜单由一条或多条菜品记录组成,每条记录一行
每条菜品记录包含:菜名、基础价格 两个信息。
订单分:点菜记录和删除信息。每一类信息都可包含一条或多条记录,每条记录一行。
点菜记录包含:序号、菜名、份额、份数。
份额可选项包括:1、2、3,分别代表小、中、大份。
删除记录格式:序号 delete
标识删除对应序号的那条点菜记录。
不同份额菜价的计算方法:
小份菜的价格=菜品的基础价格。
中份菜的价格=菜品的基础价格1.5。
小份菜的价格=菜品的基础价格2。
如果计算出现小数,按四舍五入的规则进行处理。
参考以下类的模板进行设计:
菜品类:对应菜谱上一道菜的信息。
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)//根据序号查找一条记录
}
输入格式:
菜品记录格式:
菜名+英文空格+基础价格
如果有多条相同的菜名的记录,菜品的基础价格以最后一条记录为准。
点菜记录格式:
序号+英文空格+菜名+英文空格+份额+英文空格+份数
注:份额可输入(1/2/3), 1代表小份,2代表中份,3代表大份。
删除记录格式:序号 +英文空格+delete
最后一条记录以“end”结束。
输出格式:
按顺序输出每条订单记录的处理信息,
每条点菜记录输出:序号+英文空格+菜名+英文空格+价格。其中的价格等于对应记录的菜品*份数,序号是之前输入的订单记录的序号。
如果订单中包含不能识别的菜名,则输出“** does not exist”,**是不能识别的菜名
如果删除记录的序号不存在,则输出“delete error”
最后输出订单上所有菜品的总价(整数数值),
本次题目不考虑其他错误情况,如:菜单订单顺序颠倒、不符合格式的输入、序号重复等。
输入样例:
在这里给出一组输入。例如:
麻婆豆腐 12
油淋生菜 9
1 麻婆豆腐 2 2
2 油淋生菜 1 3
end
输出样例:
在这里给出相应的输出。例如:
1 麻婆豆腐 36
2 油淋生菜 27
63
输入样例1:
订单中包含删除记录。例如:
麻婆豆腐 12
油淋生菜 9
1 麻婆豆腐 2 2
2 油淋生菜 1 3
1 delete
end
输出样例1:
在这里给出相应的输出。例如:
1 麻婆豆腐 36
2 油淋生菜 27
27
输入样例2:
订单中包含不存在的菜品记录。例如:
麻婆豆腐 12
油淋生菜 9
1 麻婆豆腐 2 2
2 油淋生菜 1 3
3 麻辣鸡丝 1 2
end
输出样例2:
在这里给出相应的输出。例如:
1 麻婆豆腐 36
2 油淋生菜 27
麻辣鸡丝 does not exist
63
输入样例3:
订单中包含删除信息以及不存在的菜品记录。例如:
麻婆豆腐 12
油淋生菜 9
1 麻婆豆腐 2 2
2 油淋生菜 1 3
3 麻辣鸡丝 1 2
1 delete
7 delete
end
输出样例3:
在这里给出相应的输出。例如:
1 麻婆豆腐 36
2 油淋生菜 27
麻辣鸡丝 does not exist
delete error;
27
输入样例4:
订单中包含删除信息以及不存在的菜品记录。例如:
麻婆豆腐 12
油淋生菜 9
1 麻婆豆腐 2 2
2 油淋生菜 1 3
3 麻辣鸡丝 1 2
5 delete
7 delete
end
输出样例4:
在这里给出相应的输出。例如:
1 麻婆豆腐 36
2 油淋生菜 27
麻辣鸡丝 does not exist
delete error;
delete error;
63
import java.util.ArrayList; import java.util.Scanner; class Dish { String name; int unit_price; public Dish(String name, int unit_price) { this.name = name; this.unit_price = unit_price; } int getPrice(int portion) { double price; if (portion == 1) { price = unit_price; } else if (portion == 2) { price = unit_price * 1.5; } else { price = unit_price * 2; } return (int) Math.round(price); } } class Menu { ArrayList<Dish> dishes; public Menu() { dishes = new ArrayList<>(); } Dish searchDish(String dishName) { for (Dish dish : dishes) { if (dish.name.equals(dishName)) { return dish; } } return null; } void addDish(String dishName, int unit_price) { Dish dish = searchDish(dishName); if (dish == null) { dish = new Dish(dishName, unit_price); dishes.add(dish); } else { dish.unit_price = unit_price; } } } class Record { int orderNum; Dish dish; int portion; int num; public Record(int orderNum, Dish dish, int portion, int num) { this.orderNum = orderNum; this.dish = dish; this.portion = portion; this.num = num; } int getPrice() { return dish.getPrice(portion) * num; } } class Order { ArrayList<Record> records; public Order() { records = new ArrayList<>(); } int getTotalPrice() { int totalPrice = 0; for (Record record : records) { totalPrice += record.getPrice(); } return totalPrice; } Record addARecord(int orderNum, String dishName, int portion, int num, Menu menu) { Dish dish = menu.searchDish(dishName); if (dish == null) { System.out.println(dishName + " does not exist"); return null; } Record record = new Record(orderNum, dish, portion, num); records.add(record); return record; } void delARecordByOrderNum(int orderNum) { for (int i = 0; i < records.size(); i++) { if (records.get(i).orderNum == orderNum) { records.remove(i); return; } } System.out.println("delete error"); } Record findRecordByNum(int orderNum) { for (Record record : records) { if (record.orderNum == orderNum) { return record; } } return null; } } public class Main { public static void main(String[] args) { Menu menu = new Menu(); Order order = new Order(); 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 unit_price = Integer.parseInt(parts[1]); menu.addDish(dishName, unit_price); } else if (parts.length == 4) { int orderNum = Integer.parseInt(parts[0]); if (parts[1].equals("delete")) { order.delARecordByOrderNum(orderNum); } else { String dishName = parts[1]; int portion = Integer.parseInt(parts[2]); int num = Integer.parseInt(parts[3]); order.addARecord(orderNum, dishName, portion, num, menu); } } } for (Record record : order.records) { int price = record.getPrice(); System.out.println(record.orderNum + " " + record.dish.name + " " + price); } int totalPrice = order.getTotalPrice(); System.out.println(totalPrice); } }
菜单计价程序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.text.ParseException; import java.text.SimpleDateFormat; import java.util.*; public class Main { public static void main(String[] args) throws ParseException { Scanner input = new Scanner(System.in); int i, t = 0, year, month, day, shi, fen, miao, f = 0, y = -1; String a = null; Menu menu = new Menu(); Table[] tables = new Table[10]; for (i = 0; ; i++) { if(f==1) break; if (y == -1) a = input.nextLine(); if (a.equals("end")) break; String[] s = a.split(" "); int x = s.length; if (x <= 3 && t == 0) {//菜品 int l = 0; int n = 0; if (!s[1].matches("[1-9]||[1-9][0-9]||[1-2][0-9][0-9]")) { System.out.println("wrong format"); menu.dishs[menu.t] = new Dish(); } else { n = Integer.parseInt(s[1]); boolean special = false; if (x == 3) { if(s[2].matches("T")) special = true; else{ System.out.println("wrong format"); System.exit(0); } } menu.addDish(s[0], n, special); if (n <= 0 || n >= 300) {//菜价超出范围 System.out.println(s[0] + " price out of range " + n); } } } else { t = 1; if(x>4){ System.out.println("wrong format"); System.exit(0); } while (true) { if(f==1) break; y++; if (x == 4 && !(s[0].matches("table")) && y == 0) {//第一个 System.out.println("wrong format"); System.exit(0); } while ((x == 4 && s[0].matches("table")) || y > 0) { if(f==1) break; if (s.length == 4) {////后面的桌子直接进入点菜,后面所有的信息并入上一桌一起计算 tables[y] = new Table(); tables[y].order = new Order(); String[] s1 = s[2].split("/");//年月日的分割 String[] s2 = s[3].split("/");//时分秒的分割 SimpleDateFormat d = new SimpleDateFormat("yyyy-MM-dd"); SimpleDateFormat d2 = new SimpleDateFormat("u"); Date date = d.parse(s1[0] + "-" + s1[1] + "-" + s1[2]);//日期格式化 int week = Integer.parseInt(d2.format(date));//提取星期几 tables[y].week = week; if (!s[1].matches("\\d*")) {//检查"table"部分 System.out.println("wrong format"); if (y == 0)//第一个桌子桌号错误直接退出 System.exit(0); else//后面所有的信息并入上一桌一起计算 y--; } else { tables[y].num = Integer.parseInt(s[1]); tables[y].year = Integer.parseInt(s1[0]); tables[y].month = Integer.parseInt(s1[1]); tables[y].day = Integer.parseInt(s1[2]); tables[y].shi = Integer.parseInt(s2[0]); tables[y].fen = Integer.parseInt(s2[1]); tables[y].miao = Integer.parseInt(s2[2]); if(s[1].matches("[0].+")){ System.out.println("wrong format"); System.exit(0); } if (!(tables[y].num <= 55 && tables[y].num >= 1)) { System.out.println("table num out of range"); System.exit(0); } else if(!tables[y].check()){ System.out.println(tables[y].num+" date error"); System.exit(0); } else if(tables[y].year>2023||tables[y].year<2022){ System.out.println("not a valid time period"); System.exit(0); } else System.out.println("table " + Integer.parseInt(s[1]) + ": "); } } else { System.out.println("wrong format"); f = 0; y--;//数据并入上一卓 } for (; ; i++) {//点菜 if (f == 1 || f == 2) break; String aa = input.nextLine(); String[] ss = aa.split(" "); //System.out.println(y + "---------" + aa); if (ss.length == 4 && ss[0].charAt(0) == 't') {//新桌子 s = ss; y++; break; } if (ss.length == 4) {//点菜 //System.out.println(y+"%"+ss[0]); //tables[y].order.addARecord(Integer.parseInt(ss[0]), ss[1], Integer.parseInt(ss[2]), Integer.parseInt(ss[3]), menu, tables[y].week); if (!tables[y].checkorder(Integer.parseInt(ss[0])))//检查订单序号顺序 System.out.println("record serial number sequence error"); else if (menu.searthDish(ss[1])==null)//订单的菜不存在 System.out.println(ss[1] + " does not exist"); else if (Integer.parseInt(ss[2]) > 3) {//份额超过3 System.out.println(ss[0] + " " + "portion out of range" + " " + ss[2]); } else if (Integer.parseInt(ss[3]) > 15) {//订单大于15 System.out.println(ss[0] + " " + "num out of range" + " " + ss[3]); } else { tables[y].order.addARecord(Integer.parseInt(ss[0]), ss[1], Integer.parseInt(ss[2]), Integer.parseInt(ss[3]), menu, tables[y].week); System.out.println(ss[0] + " " + ss[1] + " " + tables[y].order.records[tables[y].order.s - 1].ygetPrice()); tables[y].order.records[tables[y].order.s - 1].shi=tables[y].shi; tables[y].order.records[tables[y].order.s - 1].fen=tables[y].fen; } } if (ss.length == 2 && ss[1].matches("delete")) {//删除 if(tables[y].order.findRecordByNum(Integer.parseInt(ss[0]))==-1) { System.out.println("delete error"); } else { tables[y].order.delARecordByOrderNum(Integer.parseInt(ss[0])); if (tables[y].order.records[tables[y].order.findRecordByNum(Integer.parseInt(ss[0]))].life == 2) //重复删除的情况 System.out.println("deduplication " + ss[0]); } } if ((ss.length == 2 || ss.length == 3) && !ss[1].matches("delete")) { System.out.println("invalid dish"); } if (ss.length > 4) { a = aa; s = a.split(" "); f = 2; } // System.out.println("////"); /*if(ss.length==5){//代点菜 for(int t=0;t<=y;t++){ if(ss[0].equals(tables[i].num)) } }*/ if (aa.equals("end")) { f = 1; break; } } if (f == 1 || f == 2) break; } if (f == 1) break; } } } for (i = 0; i <= y; i++) { System.out.println("table " + tables[i].num + ": " + tables[i].order.ygetTotalPrice() + " " + tables[i].order.getTotalPrice()); } } } class Dish { String name;//菜品名称 int unit_price; //单价 int life=0;// boolean special = false; public Dish(){ } public Dish(String name,int unit_price,boolean special){ this.name = name; this.unit_price = unit_price; this.special = special; } int getPrice(int portion){ int s = 0; if(portion==1) s = unit_price*1; if (portion==2) s = (int) Math.round(unit_price*1.5); if(portion==3) s = unit_price*2; return s; }//计算菜品价格的方法,输入参数是点菜的份额(输入数据只能是1/2/3,代表小/中/大份) } class Menu { Dish[] dishs = new Dish[10];//菜品数组,保存所有菜品信息 int t=0; public Menu(){ } Dish searthDish(String dishName) { for (int i = 0;i < t; i++) { if (dishName.equals(dishs[i].name)) { return dishs[i]; } } return null; } //根据菜名在菜谱中查找菜品信息,返回Dish对象。 Dish addDish(String dishName,int unit_price,boolean special){ if(t>0){ int k=t-1; for (;k>=0;k--){ if (dishName.matches(dishs[k].name)){ dishs[k].unit_price = unit_price; return null; } } } dishs[t] = new Dish(dishName,unit_price,special); t++; return dishs[t-1]; }//添加一道菜品信息 } class Record { int orderNum;//序号 Dish d;//菜品\\ int portion;//份额(1/2/3代表小/中/大份) int num;//数量 int week;//星期几 int shi; int fen; int miao; int life = 0;//初始为0,删除为1,重复删除为2,无视为3 public Record(){ } int getPrice(){ int s=1; if(d.special==true) {//特价菜订单价格 if (week <= 5 && week >= 1) s = (int)Math.round(d.getPrice(portion) * num * 0.7); else s = (int) Math.round(d.getPrice(portion) * num); } else{//普通菜订单价格 if(week <= 5 && week >= 1) { if ((shi >= 17 && shi < 20) || (shi == 20 && fen <= 30)) s = (int) Math.round(d.getPrice(portion) * num * 0.8); else s = (int) Math.round(d.getPrice(portion) * num * 0.6); } else s = (int) Math.round(d.getPrice(portion) * num); } return s; }//计价,计算本条记录的价格 int ygetPrice(){ int s=1; s = (int) Math.round(d.getPrice(portion) * num); return s; } } class Order { Record[] records = new Record[100];//保存订单上每一道的记录 int i=0,s=0;//s为点菜数 int year; int month; int day; public Order(){ } int getTotalPrice(){ int num = 0; for(i=0;i<s;i++){ if(records[i].life==0) num = num + records[i].getPrice(); } return num; }//计算订单的总价 int ygetTotalPrice(){ int num = 0; for(i=0;i<s;i++){ if(records[i].life==0) num = num + records[i].ygetPrice(); } return num; }//计算订单的原总价 public Record addARecord(int orderNum,String dishName,int portion,int num,Menu menu,int week){ for (i=0;i<s;i++){ if(dishName==records[i].d.name&&portion==records[i].portion) { records[i].num = records[i].num + num; s++; return records[i]; } } if(i==s) { records[i] = new Record(); if(menu.searthDish(dishName)!=null) { records[i].d = menu.searthDish(dishName); records[i].orderNum = orderNum; records[i].d.name = dishName; records[i].portion = portion; records[i].num = num; records[i].week = week; s++; } else { records[i].d = new Dish(); records[i].d.name=dishName; records[i].life=6;//订单的菜不存在 s++; } return records[i]; } return null; }//添加一条菜品信息到订单中。 public void delARecordByOrderNum(int orderNum){ for (i=0;i<s;i++){ if(records[i].orderNum==orderNum){ if(records[i].life==1) records[i].life = 2; else records[i].life = 1; } } }//根据序号删除一条记录 public int findRecordByNum(int orderNum){ for (i=0;i<s;i++){ if (records[i].orderNum==orderNum) return i; } return -1; }//根据序号查找一条记录 } class Table { Order order; int num; int year; int month; int day; int shi; int fen; int miao; int week; public Table(){ } public boolean check(){ int[] days = {31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31}; if (month >= 1 && month <= 12) { //判断是否为闰年 if ((year % 100 == 0 && year % 400 == 0) || year % 4 == 0) { //判断当前月份是否为2月,因为闰年的2月份为29天 if (month == 2 && day <= 29) return true; else { if (day <= days[month - 1]) return true; } } else { if (day <= days[month - 1]) return true; } } return false; } boolean checkorder(int x){//检查点菜序号 for(int j=0;j<=order.s-1;j++){ if(x<=order.records[j].orderNum) return false; } return true; } }
菜单计价程序5
本题在菜单计价程序-3的基础上增加了部分内容,增加的内容用加粗字体标识。
注意不是菜单计价程序-4,本题和菜单计价程序-4同属菜单计价程序-3的两个不同迭代分支。
设计点菜计价程序,根据输入的信息,计算并输出总价格。
输入内容按先后顺序包括两部分:菜单、订单,最后以"end"结束。
菜单由一条或多条菜品记录组成,每条记录一行
每条菜品记录包含:菜名、基础价格 三个信息。
订单分:桌号标识、点菜记录和删除信息、代点菜信息。每一类信息都可包含一条或多条记录,每条记录一行或多行。
桌号标识独占一行,包含两个信息:桌号、时间。
桌号以下的所有记录都是本桌的记录,直至下一个桌号标识。
点菜记录包含:序号、菜名、份额、份数。份额可选项包括:1、2、3,分别代表小、中、大份。
不同份额菜价的计算方法:小份菜的价格=菜品的基础价格。中份菜的价格=菜品的基础价格1.5。小份菜的价格=菜品的基础价格2。如果计算出现小数,按四舍五入的规则进行处理。
删除记录格式:序号 delete
标识删除对应序号的那条点菜记录。
如果序号不对,输出"delete error"
代点菜信息包含:桌号 序号 菜品名称 口味度 份额 份数
代点菜是当前桌为另外一桌点菜,信息中的桌号是另一桌的桌号,带点菜的价格计算在当前这一桌。
程序最后按输入的先后顺序依次输出每一桌的总价(注意:由于有代点菜的功能,总价不一定等于当前桌上的菜的价格之和)。
每桌的总价等于那一桌所有菜的价格之和乘以折扣。如存在小数,按四舍五入规则计算,保留整数。
折扣的计算方法(注:以下时间段均按闭区间计算):
周一至周五营业时间与折扣:晚上(17:00-20:30)8折,周一至周五中午(10:30--14:30)6折,其余时间不营业。
周末全价,营业时间:9:30-21:30
如果下单时间不在营业范围内,输出"table " + t.tableNum + " out of opening hours"
参考以下类的模板进行设计:菜品类:对应菜谱上一道菜的信息。
Dish {
String name;//菜品名称
int unit_price; //单价
int getPrice(int portion)//计算菜品价格的方法,输入参数是点菜的份额(输入数据只能是1/2/3,代表小/中/大份) }
菜谱类:对应菜谱,包含饭店提供的所有菜的信息。
Menu {
Dish[] dishs ;//菜品数组,保存所有菜品信息
Dish searthDish(String dishName)//根据菜名在菜谱中查找菜品信息,返回Dish对象。
Dish addDish(String dishName,int unit_price)//添加一道菜品信息
}
点菜记录类:保存订单上的一道菜品记录
Record {
int orderNum;//序号\\
Dish d;//菜品\\
int portion;//份额(1/2/3代表小/中/大份)\\
int getPrice()//计价,计算本条记录的价格\\
}
订单类:保存用户点的所有菜的信息。
Order {
Record[] records;//保存订单上每一道的记录
int getTotalPrice()//计算订单的总价
Record addARecord(int orderNum,String dishName,int portion,int num)//添加一条菜品信息到订单中。
delARecordByOrderNum(int orderNum)//根据序号删除一条记录
findRecordByNum(int orderNum)//根据序号查找一条记录
}
### 输入格式:
桌号标识格式:table + 序号 +英文空格+ 日期(格式:YYYY/MM/DD)+英文空格+ 时间(24小时制格式: HH/MM/SS)
菜品记录格式:
菜名+英文空格+基础价格
如果有多条相同的菜名的记录,菜品的基础价格以最后一条记录为准。
点菜记录格式:序号+英文空格+菜名+英文空格+份额+英文空格+份数注:份额可输入(1/2/3), 1代表小份,2代表中份,3代表大份。
删除记录格式:序号 +英文空格+delete
代点菜信息包含:桌号+英文空格+序号+英文空格+菜品名称+英文空格+份额+英文空格+分数
最后一条记录以“end”结束。
### 输出格式:
按输入顺序输出每一桌的订单记录处理信息,包括:
1、桌号,格式:table+英文空格+桌号+”:”
2、按顺序输出当前这一桌每条订单记录的处理信息,
每条点菜记录输出:序号+英文空格+菜名+英文空格+价格。其中的价格等于对应记录的菜品\*份数,序号是之前输入的订单记录的序号。如果订单中包含不能识别的菜名,则输出“\*\* does not exist”,\*\*是不能识别的菜名
如果删除记录的序号不存在,则输出“delete error”
最后按输入顺序一次输出每一桌所有菜品的总价(整数数值)格式:table+英文空格+桌号+“:”+英文空格+当前桌的总价
以上为菜单计价系列-3的题目要求,加粗的部分是有调整的内容。本次课题相比菜单计价系列-3新增要求如下:
1、菜单输入时增加特色菜,特色菜的输入格式:菜品名+英文空格+口味类型+英文空格+基础价格+"T"
例如:麻婆豆腐 川菜 9 T
菜价的计算方法:
周一至周五 7折, 周末全价。
特色菜的口味类型:川菜、晋菜、浙菜
川菜增加辣度值:辣度0-5级;对应辣度水平为:不辣、微辣、稍辣、辣、很辣、爆辣;
晋菜增加酸度值,酸度0-4级;对应酸度水平为:不酸、微酸、稍酸、酸、很酸;
浙菜增加甜度值,甜度0-3级;对应酸度水平为:不甜、微甜、稍甜、甜;
例如:麻婆豆腐 川菜 9 T
输入订单记录时如果是特色菜,添加口味度(辣/酸/甜度)值,格式为:序号+英文空格+菜名+英文空格+口味度值+英文空格+份额+英文空格+份数
例如:1 麻婆豆腐 4 1 9
单条信息在处理时,如果口味度超过正常范围,输出"spicy/acidity/sweetness num out of range : "+口味度值,spicy/acidity/sweetness(辣度/酸度/甜度)根据菜品类型择一输出,例如:
acidity num out of range : 5
输出一桌的信息时,按辣、酸、甜度的顺序依次输出本桌菜各种口味的口味度水平,如果没有某个类型的菜,对应的口味(辣/酸/甜)度不输出,只输出已点的菜的口味度。口味度水平由口味度平均值确定,口味度平均值只综合对应口味菜系的菜计算,不做所有菜的平均。比如,某桌菜点了3份川菜,辣度分别是1、3、5;还有4份晋菜,酸度分别是,1、1、2、2,辣度平均值为3、酸度平均值四舍五入为2,甜度没有,不输出。
一桌信息的输出格式:table+英文空格+桌号+:+英文空格+当前桌的原始总价+英文空格+当前桌的计算折扣后总价+英文空格+"川菜"+数量+辣度+英文空格+"晋菜"+数量+酸度+英文空格+"浙菜"+数量+甜度。
如果整桌菜没有特色菜,则只输出table的基本信息,格式如下,注意最后加一个英文空格:
table+英文空格+桌号+:+英文空格+当前桌的原始总价+英文空格+当前桌的计算折扣后总价+英文空格
例如:table 1: 60 36 川菜 2 爆辣 浙菜 1 微甜
计算口味度时要累计本桌各类菜系所有记录的口味度总和(每条记录的口味度乘以菜的份数),再除以对应菜系菜的总份数,最后四舍五入。
注:本题要考虑代点菜的情况,当前桌点的菜要加上被其他桌代点的菜综合计算口味度平均值。
2、考虑客户订多桌菜的情况,输入时桌号时,增加用户的信息:
格式:table+英文空格+桌号+英文空格+":"+英文空格+客户姓名+英文空格+手机号+日期(格式:YYYY/MM/DD)+英文空格+ 时间(24小时制格式: HH/MM/SS)
例如:table 1 : tom 13670008181 2023/5/1 21/30/00
约束条件:客户姓名不超过10个字符,手机号11位,前三位必须是180、181、189、133、135、136其中之一。
输出结果时,先按要求输出每一桌的信息,最后按字母顺序依次输出每位客户需要支付的金额。不考虑各桌时间段的问题,同一个客户的所有table金额都要累加。
输出用户支付金额格式:
用户姓名+英文空格+手机号+英文空格+支付金额
注意:不同的四舍五入顺序可能会造成误差,请按以下步骤累计一桌菜的菜价:
计算每条记录的菜价:将每份菜的单价按份额进行四舍五入运算后,乘以份数计算多份的价格,然后乘以折扣,再进行四舍五入,得到本条记录的最终支付价格。
将所有记录的菜价累加得到整桌菜的价格。
输入格式:
桌号标识格式:table + 序号 +英文空格+ 日期(格式:YYYY/MM/DD)+英文空格+ 时间(24小时制格式: HH/MM/SS)
菜品记录格式:
菜名+口味类型+英文空格+基础价格
如果有多条相同的菜名的记录,菜品的基础价格以最后一条记录为准。
点菜记录格式:序号+英文空格+菜名+英文空格+辣/酸/甜度值+英文空格+份额+英文空格+份数 注:份额可输入(1/2/3), 1代表小份,2代表中份,3代表大份。辣/酸/甜度取值范围见题目中说明。
删除记录格式:序号 +英文空格+delete
代点菜信息包含:桌号+英文空格+序号+英文空格+菜品名称**+英文空格+辣/酸/甜度值+**英文空格+份额+英文空格+分数
最后一条记录以“end”结束。
输出格式:
按输入顺序输出每一桌的订单记录处理信息,包括:
1、桌号,格式:table+英文空格+桌号+“:”+英文空格
2、按顺序输出当前这一桌每条订单记录的处理信息,
每条点菜记录输出:序号+英文空格+菜名+英文空格+价格。其中的价格等于对应记录的菜品\*份数,序号是之前输入的订单记录的序号。如果订单中包含不能识别的菜名,则输出“\*\* does not exist”,\*\*是不能识别的菜名
如果删除记录的序号不存在,则输出“delete error”
之后按输入顺序一次输出每一桌所有菜品的价格(整数数值),
格式:table+英文空格+桌号+“:”+英文空格+当前桌的计算折扣后总价+英文空格+辣度平均值+英文空格+酸度平均值+英文空格+甜度平均值+英文空格
最后按拼音顺序输出每位客户(不考虑客户同名或拼音相同的情况)的支付金额,格式: 用户姓名+英文空格+手机号+英文空格+支付总金额,按输入顺序排列。
输入样例1:
桌号时间超出营业范围。例如:
麻婆豆腐 川菜 12 T
油淋生菜 9
麻辣鸡丝 10
table 1 : tom 13605054400 2023/5/1 21/30/00
1 麻婆豆腐 3 1 2
2 油淋生菜 2 1
3 麻婆豆腐 2 3 2
end
输出样例1:
在这里给出相应的输出。例如:
table 1 out of opening hours
输入样例2:
一种口味的菜品。例如:
麻婆豆腐 川菜 12 T
油淋生菜 9
麻辣鸡丝 10
table 1 : tom 13605054400 2023/5/1 20/30/00
1 麻婆豆腐 2 1 2
2 油淋生菜 2 1
3 麻婆豆腐 2 3 2
end
输出样例2:
在这里给出相应的输出。例如:
table 1:
1 麻婆豆腐 24
2 油淋生菜 14
3 麻婆豆腐 48
table 1: 86 62 川菜 4 稍辣
tom 13605054400 62
输入样例3:
辣度值超出范围。例如:
麻婆豆腐 川菜 12 T
油淋生菜 9
麻辣鸡丝 10
table 1 : tom 13605054400 2023/5/1 18/30/00
1 麻婆豆腐 6 1 2
2 油淋生菜 1 1
3 麻婆豆腐 5 3 2
end
输出样例3:
在这里给出相应的输出。例如:
table 1:
spicy num out of range :6
2 油淋生菜 9
3 麻婆豆腐 48
table 1: 57 41 川菜 2 爆辣
tom 13605054400 41
输入样例4:
同一用户对应多桌菜。例如:
麻婆豆腐 川菜 12 T
油淋生菜 9
麻辣鸡丝 10
table 1 : tom 13605054400 2023/5/1 18/30/00
1 麻婆豆腐 1 1 2
2 油淋生菜 1 1
3 麻婆豆腐 2 2 2
table 2 : tom 13605054400 2023/5/6 18/30/00
1 麻婆豆腐 2 1 2
2 麻辣鸡丝 2 2
3 麻婆豆腐 2 1 1
end
输出样例4:
在这里给出相应的输出。例如:
table 1:
1 麻婆豆腐 24
2 油淋生菜 9
3 麻婆豆腐 36
table 2:
1 麻婆豆腐 24
2 麻辣鸡丝 30
3 麻婆豆腐 12
table 1: 69 49 川菜 4 稍辣
table 2: 66 66 川菜 3 稍辣
tom 13605054400 115
输入样例5:
多用户多桌菜。例如:
东坡肉 浙菜 25 T
油淋生菜 9
蜜汁灌藕 浙菜 10 T
刀削面 晋菜 10 T
醋浇羊肉 晋菜 30 T
麻婆豆腐 川菜 12 T
麻辣鸡丝 川菜 15 T
table 1 : tom 13605054400 2023/5/6 12/30/00
1 醋浇羊肉 4 1 1
3 刀削面 1 1 3
2 东坡肉 3 2 1
4 麻辣鸡丝 2 1 1
table 2 : jerry 18100334566 2023/5/1 12/30/00
1 醋浇羊肉 1 1 2
3 麻婆豆腐 2 2 1
4 麻辣鸡丝 2 3 3
table 3 : jerry 18100334566 2023/5/1 12/30/00
1 醋浇羊肉 2 1 1
3 蜜汁灌藕 1 1 2
2 东坡肉 2 2 1
4 麻辣鸡丝 5 1 1
end
输出样例5:
在这里给出相应的输出。例如:
table 1:
1 醋浇羊肉 30
3 刀削面 30
2 东坡肉 38
4 麻辣鸡丝 15
table 2:
1 醋浇羊肉 60
3 麻婆豆腐 18
4 麻辣鸡丝 90
table 3:
1 醋浇羊肉 30
3 蜜汁灌藕 20
2 东坡肉 38
4 麻辣鸡丝 15
table 1: 113 113 川菜 1 稍辣 晋菜 4 稍酸 浙菜 1 甜
table 2: 168 118 川菜 4 稍辣 晋菜 2 微酸
table 3: 103 73 川菜 1 爆辣 晋菜 1 稍酸 浙菜 3 微甜
jerry 18100334566 191
tom 13605054400 113
输入样例6:
多用户多桌菜含代点菜。例如:
东坡肉 浙菜 25 T
油淋生菜 9
蜜汁灌藕 浙菜 10 T
刀削面 晋菜 10 T
醋浇羊肉 晋菜 30 T
麻婆豆腐 川菜 12 T
麻辣鸡丝 川菜 15 T
table 1 : tom 13605054400 2023/5/6 12/30/00
1 醋浇羊肉 4 1 1
3 刀削面 1 1 3
2 东坡肉 3 2 1
4 麻辣鸡丝 2 1 1
table 2 : jerry 18100334566 2023/5/1 12/30/00
1 1 醋浇羊肉 0 1 2
3 麻婆豆腐 2 2 1
4 麻辣鸡丝 2 3 3
table 3 : lucy 18957348763 2023/5/1 12/30/00
1 醋浇羊肉 2 1 1
3 蜜汁灌藕 1 1 2
2 东坡肉 2 2 1
4 麻辣鸡丝 5 1 1
end
输出样例6:
在这里给出相应的输出。例如:
table 1:
1 醋浇羊肉 30
3 刀削面 30
2 东坡肉 38
4 麻辣鸡丝 15
table 2:
1 table 2 pay for table 1 60
3 麻婆豆腐 18
4 麻辣鸡丝 90
table 3:
1 醋浇羊肉 30
3 蜜汁灌藕 20
2 东坡肉 38
4 麻辣鸡丝 15
table 1: 113 113 川菜 1 稍辣 晋菜 6 微酸 浙菜 1 甜
table 2: 168 118 川菜 4 稍辣
table 3: 103 73 川菜 1 爆辣 晋菜 1 稍酸 浙菜 3 微甜
jerry 18100334566 118
lucy 18957348763 73
tom 13605054400 113
输入样例7:
错误的菜品记录和桌号记录,用户丢弃。例如:
东坡肉 25 T
油淋生菜 9
table 1 : tom 136050540 2023/5/1 12/30/00
2 东坡肉 3 2 1
end
输出样例7:
在这里给出相应的输出。例如:
wrong format
wrong format
import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.*; public class Main { public static void main(String[] args) throws ParseException { Menu menu = new Menu(); // 读入菜单信息 Scanner sc = new Scanner(System.in); String menuLine = sc.nextLine(); while (!menuLine.startsWith("table")) { String[] menuInfo = menuLine.split(" "); if (menuInfo.length == 2) { String name = menuInfo[0]; int unit_price = Integer.parseInt(menuInfo[1]); if (menu.searchDish(name) == null) { menu.addDish(name, unit_price); } } else if (menuInfo.length == 4 && menuLine.endsWith("T")) { String name = menuInfo[0]; String type = menuInfo[1]; int unit_price = Integer.parseInt(menuInfo[2]); Map<String, String> map = new HashMap<String, String>() { { put("川菜", "Chuan"); put("晋菜", "Jin"); put("浙菜", "Zhe"); } }; DishType dishType = DishType.valueOf(map.get(type)); if (menu.searchDish(name) == null) { menu.addDish(name, unit_price, dishType); } } else { System.out.println("wrong format"); } menuLine = sc.nextLine(); } ArrayList<Table> tables = new ArrayList<>(); ArrayList<String> names = new ArrayList<>(); // 读入订单信息 int tableId = 0; String name = null; String phone = null; Date date = null; Date time = null; boolean legaltime = true; boolean legalformat = true; String orderLine = menuLine; while (!orderLine.equals("end")) { String[] orderInfo = orderLine.split(" "); // 解析桌号标识 if (orderLine.startsWith("table")) { legalformat = true; SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy/MM/dd"); SimpleDateFormat timeFormat = new SimpleDateFormat("HH/mm/ss"); tableId = Integer.parseInt(orderInfo[1]); name = orderInfo[3]; phone = orderInfo[4]; try { date = dateFormat.parse(orderInfo[5]); time = timeFormat.parse(orderInfo[6]); } catch (Exception e) { legalformat = false; System.out.println("wrong format"); orderLine = sc.nextLine(); continue; } String regex = "^1(80|81|89|33|35|36)\\d{8}$"; Table table = new Table(tableId, name, phone, date, time); tables.add(table); if (name.length() > 10 || !phone.matches(regex)) { legalformat = false; System.out.println("wrong format"); orderLine = sc.nextLine(); continue; } if (!names.contains(name)) { names.add(name); } if (table.getCoefficient(true) == 0) { legaltime = false; System.out.println("table " + table.tableId + " out of opening hours"); } else { System.out.println(table.printId()); } } else { if (legalformat) { int orderNum; try { orderNum = Integer.parseInt(orderInfo[0]); } catch (Exception e) { System.out.println("wrong format"); orderLine = sc.nextLine(); continue; } if (orderLine.endsWith("delete")) { if (!tables.get(tableId - 1).delRecordByOrderNum(orderNum)) { System.out.println("delete error"); } } else { if (orderInfo.length == 4) { String dishName = orderInfo[1]; int portion = Integer.parseInt(orderInfo[2]); int quantity = Integer.parseInt(orderInfo[3]); Dish dish = menu.searchDish(dishName); if (dish == null) { System.out.println(dishName + " does not exist"); orderLine = sc.nextLine(); continue; } Record record = new Record(tableId, orderNum, dish, portion, quantity); tables.get(tableId - 1).addRecord(record); if (legaltime) { System.out.println(record.print(tableId)); } } else if (orderInfo.length == 5) { String dishName = orderInfo[1]; int level = Integer.parseInt(orderInfo[2]); int portion = Integer.parseInt(orderInfo[3]); int quantity = Integer.parseInt(orderInfo[4]); Dish dish = menu.searchDish(dishName); if (dish == null) { System.out.println(dishName + " does not exist"); orderLine = sc.nextLine(); continue; } Record record = new Record(tableId, orderNum, dish, level, portion, quantity); tables.get(tableId - 1).addRecord(record); if (legaltime) { System.out.println(record.print(tableId)); } } else if (orderInfo.length == 6) { int givenId = Integer.parseInt(orderInfo[1]); String dishName = orderInfo[2]; int level = Integer.parseInt(orderInfo[3]); int portion = Integer.parseInt(orderInfo[4]); int quantity = Integer.parseInt(orderInfo[5]); Dish dish = menu.searchDish(dishName); if (dish == null) { System.out.println(dishName + " does not exist"); orderLine = sc.nextLine(); continue; } Record record1 = new Record(givenId, orderNum, dish, level, portion, quantity); Record record2 = new Record(givenId, orderNum, dish, level, 0, quantity); tables.get(tableId - 1).addRecord(record1); tables.get(givenId - 1).addRecord(record2); if (legaltime) { System.out.println(record1.print(tableId)); } } else { System.out.println("wrong format"); } } } } // 读入下一个桌号标识 orderLine = sc.nextLine(); } sc.close(); for (Table table : tables) { if (table.flag && table.getTotalPrice() != 0) { System.out.println(table.printInfo()); } } names.sort(new Comparator<String>() { @Override public int compare(String s1, String s2) { return s1.compareTo(s2); } }); for (String costumName : names) { int sum = 0; String costumPhone = null; for (Table table : tables) { if (table.name.equals(costumName)) { sum += table.getCheckedPrice(); costumPhone = table.phone; } } if (sum != 0) { System.out.println(costumName + " " + costumPhone + " " + sum); } } } } enum DishType { Chuan, Jin, Zhe, } class Dish { public String name; public int unit_price; public DishType type; public Dish(String name, int unit_price, DishType type) { this.name = name; this.unit_price = unit_price; this.type = type; } public Dish(String name, int unit_price) { this.name = name; this.unit_price = unit_price; } @Override public String toString() { return name; } } class Menu { public ArrayList<Dish> dishs = new ArrayList<>(); public Dish searchDish(String dishName) { for (Dish dish : dishs) { if (dish.name.equals(dishName)) { return dish; } } return null; } void addDish(String dishName, int unit_price) { dishs.add(new Dish(dishName, unit_price)); } void addDish(String dishName, int unit_price, DishType type) { dishs.add(new Dish(dishName, unit_price, type)); } } class Record { int orderNum; Dish dish; int portion; int quantity; int level; boolean flag; int givenId; boolean check_level() { switch (dish.type) { case Chuan: if (level > 5 || level < 0) { return false; } else { return true; } case Jin: if (level > 4 || level < 0) { return false; } else { return true; } case Zhe: if (level > 3 || level < 0) { return false; } else { return true; } default: return true; } } public Record(int givenID, int orderNum, Dish dish, int portion, int quantity) { this.orderNum = orderNum; this.dish = dish; this.portion = portion; this.quantity = quantity; this.level = -1; this.flag = true; this.givenId = givenID; } public Record(int givenId, int orderNum, Dish dish, int level, int portion, int quantity) { this.orderNum = orderNum; this.dish = dish; this.portion = portion; this.quantity = quantity; this.level = level; this.flag = check_level(); this.givenId = givenId; } int getPrice() { if (!flag) return 0; double coefficient = 0; switch (portion) { case 1: coefficient = 1; break; case 2: coefficient = 1.5; break; case 3: coefficient = 2; break; } int price = (int) Math.round(dish.unit_price * coefficient) * quantity; return price; } int getCheckedPrice(Double coefficient) { return (int) Math.round(getPrice() * coefficient); } public String print(int tableId) { if (flag == false) { switch (dish.type) { case Chuan: return "spicy num out of range :" + level; case Jin: return "acidity num out of range :" + level; case Zhe: return "sweetness num out of range :" + level; default: return null; } } else { if (givenId == tableId) { return orderNum + " " + dish.toString() + " " + getPrice(); } return orderNum + " table " + tableId + " pay for table " + givenId + " " + getPrice(); } } @Override public String toString() { return "Record [orderNum=" + orderNum + ", dish=" + dish + ", portion=" + portion + ", quantity=" + quantity + ", level=" + level + ", flag=" + flag + ", givenId=" + givenId + "]"; } } class Table { ArrayList<Record> records = new ArrayList<>(); int tableId; String name; String phone; Date date; Date time; boolean flag; public Table(int tableId, String name, String phone, Date date, Date time) { this.name = name; this.phone = phone; this.date = date; this.time = time; this.tableId = tableId; this.flag = true; } double getCoefficient(boolean Special) throws ParseException { double coefficient = 0; SimpleDateFormat sdfTime = new SimpleDateFormat("HH:mm"); Calendar cal = Calendar.getInstance(); cal.setTime(date); int dayOfWeek = cal.get(Calendar.DAY_OF_WEEK); // 营业时间 if (dayOfWeek == 1 || dayOfWeek == 7) { if (time.after(sdfTime.parse("9:29")) && time.before(sdfTime.parse("21:31"))) { coefficient = 1; } } else { if (time.after(sdfTime.parse("16:59")) && time.before(sdfTime.parse("20:31"))) { if (Special) { coefficient = 0.7; } else { coefficient = 0.8; } } else if (time.after(sdfTime.parse("10:29")) && time.before(sdfTime.parse("14:31"))) { if (Special) { coefficient = 0.7; } else { coefficient = 0.6; } } } if (coefficient == 0) { flag = false; } return coefficient; } int getTotalPrice() { int sum = 0; for (Record record : records) { sum += record.getPrice(); } return sum; } int getCheckedPrice() throws ParseException { int sum = 0; for (Record record : records) { if (record.level != -1) { sum += record.getCheckedPrice(getCoefficient(true)); } else { sum += record.getCheckedPrice(getCoefficient(false)); } } return sum; } String getAveLevel(DishType type) { String[] spicy = { "不辣", "微辣", "稍辣", "辣", "很辣", "爆辣" }; String[] acidity = { "不酸", "微酸", "稍酸", "酸", "很酸" }; String[] sweetness = { "不甜", "微甜", "稍甜", "甜" }; double sum = 0; double num = 0; for (Record record : records) { if (record.dish.type == type) { if (record.flag && tableId == record.givenId) { num += record.quantity; sum += record.level * record.quantity; } } } if (num == 0) { return ""; } int ave = (int) Math.round(sum / num); switch (type) { case Chuan: return " 川菜 " + (int) num + " " + spicy[ave]; case Jin: return " 晋菜 " + (int) num + " " + acidity[ave]; case Zhe: return " 浙菜 " + (int) num + " " + sweetness[ave]; default: return null; } } void addRecord(Record record) { records.add(record); } boolean delRecordByOrderNum(int orderNum) { return records.removeIf(record -> record.orderNum == orderNum); } Record findRecordByOrderNum(int orderNum) { for (Record record : records) { if (record.orderNum == orderNum) { return record; } } return null; } public String printId() { return "table " + tableId + ": "; } public String printInfo() throws ParseException { String chuan = getAveLevel(DishType.Chuan); String jin = getAveLevel(DishType.Jin); String zhe = getAveLevel(DishType.Zhe); if (chuan == "" && jin == "" && zhe == "") { return "table " + tableId + ": " + getTotalPrice() + " " + getCheckedPrice() + " "; } else { return "table " + tableId + ": " + getTotalPrice() + " " + getCheckedPrice() + chuan + jin + zhe; } } }
三、踩坑心得
其实第一次作业考察的内容相对来说比较简单,但是让我踩坑的是刚开始九道题我只有一道是完全正确的,其他的八道题目都有测试点没过,非常难受,并且绞尽脑汁觉得自己的代码没有任何的问题,但实际上是被一个小细节给害惨了,把我所有题目中定义变量的的double全部改为了float,测试点大部分就过去了,真的是人生无常,大肠包小肠~ 吸取教训,正确地运用并把握审题精度是处理细节非常重要的手段之一。
还有第三次作业真的三道题目一个大连贯,从点到线再到形类,初步推测应该是需要设计的类变多了,分割字符串,并且从最初的两个点的坐标到三个点以及四个点五个点的坐标,一步步地复杂化,但主要地一些基础还是没有变化的。老师上课讲了正则表达式,但没有认真的听,所以踩坑了。
四、改进建议
对第二次作业我认为我函数内的代码行数还是太长,而且不是很清晰明了,希望自己以后条件判断还是多用switch case的条件语句,少用if else语句,让自己的代码变得更可读一些。类似的性质也可以做成类函数,就不会让代码那么长一串,用相应的英文函数名表示就明白多了。
对第三次作业我认为我需要去更多的了解类的设计和有关正则的方式。我在最后一题使用的合法性判断很杂是混合在各个功能中的,这样不仅大程度的提高了代码的复杂度而且可读性很低,大概只有我自己能弄的清楚的程度。老师在课上介绍了正则的方法去限定合法性,我还需要对其更深的了解直到能够熟练运用。具体要做的改进首先是用正则的方法写合法测试,再将多次项的分类作为一个类,此类中含整数项判断函数、一次项判断函数和多次项判断函数,主函数在进行筛选时就可以直接调用类中的函数,这样可以大幅的的减少重复代码,减少查重降低复杂度。
五、总结
目前,我认为自己在Java方法的运用中仍存在不足,对其运用的具体场景和条件以及具体的内部原理了解不够透彻,仍需要进一步学习和实践。除此之外,我对于多个类的程序设计仍比较陌生,需要更进一步的学习。另外,自己在编程过程中需要养成更好的习惯,首先要习惯用面向对象的思维去思考问题、设计算法;其次,代码的书写要规范、美观、简略,这也方便自己查找错误,避免了很多不必要的麻烦。对老师就是希望老师能把上课用于讲课的那个知识点文件发给我让我瞅瞅研究研究,然后课程实验作业什么的能够更多的要求类上面的设计以及可读程度。三次作业逐步引导我从面向对象程序设计的思维过渡到面向对象的程序设计思维我认为Java中的类和C语言中的函数有些联系,Java中的类就好像把好几个函数包在一个大函数里,然后主函数可以去调用这个大函数,但是它们之间的区别又是很明显的Java中的类是有性质和方法的是可以拥有实体的,C语言中的函数并不能。还有Java在多人合作项目中确实更方便,因为每个类都可以拥有它们自身的变量,而且可以分好几个文件运行只要在一个包下就行,而且Java类如果写的好的话就很容易让别人去解读。所以我觉得在Java的学习中要学会对类的合理设计,让自己的代码变得看起来舒服可读性高。而且我觉得老师可以在布置pta作业的时候少量多次,从简到易,逐步深入,时长也可以缩短一些。