一、前言:
(1)知识点
本次作业,知识点很广,单论期中考试,就涉及了许多,类结构设计、继承与多态和抽象类与接口等。
Java类结构知识点:
-
类的声明:Java类通过关键字 "class" 来声明,后面跟着类名和类体,类体包含类的成员变量和方法。
-
成员变量:也称为字段或属性,用于描述类的状态。它们定义了类的特征和行为。
-
方法:类中的函数被称为方法,用于定义类的行为和操作。方法包括构造方法、实例方法和静态方法。
-
构造方法:用于创建对象时进行初始化操作的特殊方法。构造方法的名称与类名相同,不返回任何类型的值。
-
访问修饰符:可以用来限定类的成员变量和方法的访问权限,常用的有 public、protected、private 和默认访问权限(即不加修饰符)。
-
封装:通过将类的属性私有化,并提供公共的方法来访问或修改这些属性,从而控制对类的访问,这就是封装。
-
继承:子类可以继承父类的属性和方法,从而实现代码重用和层次化组织类的结构。
-
多态:允许不同类的对象对同一消息作出响应,实现了接口重用性和灵活性。
-
抽象类和接口:抽象类是不能被实例化的类,通常用于被继承;接口定义了一系列方法的集合,子类可以实现接口中定义的方法。
-
内部类:在一个类的内部定义的类,可以分为成员内部类、静态内部类、局部内部类和匿名内部类。
Java继承和多态的知识点:
继承是指一个类(子类)可以从另一个类(父类)那里获得属性和方法的过程。
特点:
- 子类会继承父类的非私有字段和方法,可以重用父类的代码。
- 子类可以增加新的字段和方法,也可以重写父类的方法。
- 通过继承,可以实现代码的层次化组织和提高代码的复用性。
多态是指同一个方法调用在不同的对象上可以有不同的行为。
- 实现方式:通过方法重载(Overloading)和方法重写(Overriding)来实现多态。
- 方法重载:在同一个类中,可以定义多个方法名相同但参数列表不同的方法,编译器根据方法的参数类型和数量来决定调用哪个方法。
- 方法重写:子类可以重写父类的方法,当子类对象调用这个方法时,将执行子类中的方法而不是父类中的方法。
- 多态的好处:
- 提高了代码的灵活性,使得程序可以更容易地应对不同的对象。
- 可以实现接口重用性,允许不同类的对象对同一消息作出响应。
(2)题量
本次写了PTA题目集4、5、6以及期中考试关于圆类设计,类结构设计,继承与多态和抽象类与接口等。题量适中,在可行范围内。
(3)难度
难度这块,首先PTA上面,难度成阶梯式上升,但是当到中间完成了菜单-4之后,再写后面的,又不是特别难。期中考试是四道大题,难度还算可以,按照规则并联系平时所学,还是可以在规定时间内正确完成的。
二、设计与分析
题目集4,7-4
题目:7-4 菜单计价程序-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
代码长度限制
16 KB
时间限制
400 ms
内存限制
64 MB
代码如下:
import java.util.Scanner; class Dish { String name; int unitPrice; int num; int getPrice(int portion, int num) { int price = 0; switch (portion) { case 1: price = unitPrice * num; break; case 2: price = Math.round(unitPrice * 1.5f) * num; break; case 3: price = unitPrice * 2 * num; break; } return price; } } class Menu { Dish[] dishes; int count = 0; Menu(int capacity) { dishes = new Dish[capacity]; } Dish searchDish(String dishName) { for (int i = count - 1; i >= 0; i--) { if (dishName.equals(dishes[i].name)) { return dishes[i]; } } return null; } Dish addDish(String dishName, int unitPrice) { Dish dish = new Dish(); dish.name = dishName; dish.unitPrice = unitPrice; dishes[count++] = dish; return dish; } } class Order { static class Record { Dish dish; int portion; int num; int getPrice() { return dish.getPrice(portion, num); } } Record[] records; boolean[] deleted; int count = 0; Order(int capacity) { records = new Record[capacity]; deleted = new boolean[capacity]; } void addRecord(Dish dish, int portion, int num) { Record record = new Record(); record.dish = dish; record.portion = portion; record.num = num; records[count++] = record; dish.num += num; } boolean deleteRecordByIndex(int index) { if (index >= 0 && index < count && !deleted[index]) { for (int i = count - 1; i >= index + 1; i--) { records[i - 1] = records[i]; } records[count - 1] = null; deleted[index] = true; count--; return true; } else { return false; } } int getTotalPrice() { int sum = 0; for (int i = 0; i < count; i++) { sum += records[i].getPrice(); } return sum; } } public class Main { public static void main(String[] args) { Scanner scanner = new Scanner(System.in); Menu menu = new Menu(10); Order order = new Order(1000); while (true) { String input = scanner.nextLine(); if (input.equals("end")) { break; } String[] temp = input.split(" "); if (temp.length == 2 && temp[1].equals("delete")) { int index = Integer.parseInt(temp[0]) - 1; if (!order.deleteRecordByIndex(index)) { System.out.println("delete error;"); } } else if (temp.length == 4) { int orderNum = Integer.parseInt(temp[0]); String dishName = temp[1]; int portion = Integer.parseInt(temp[2]); int num = Integer.parseInt(temp[3]); Dish dish = menu.searchDish(dishName); if (dish == null) { System.out.println(dishName + " does not exist"); } else { order.addRecord(dish, portion, num); System.out.println(orderNum + " " + dishName + " " + order.records[order.count - 1].getPrice()); } } else if (temp.length == 2) { String dishName = temp[0]; int unitPrice = Integer.parseInt(temp[1]); menu.addDish(dishName, unitPrice); } } System.out.print(order.getTotalPrice()); } }
自我分析:
代码包括以下几个类:
Dish 类表示菜品,包含菜品名称、单价、数量和计算价格的方法。
Menu 类表示菜单,包含菜品数组和数量,提供了查找菜品和添加菜品的方法。
Order 类表示订单,包含订单记录数组和删除标记数组,提供了添加订单记录、删除指定索引的订单记录和计算订单总价的方法。
Main 类是主程序入口,实现了整个点餐系统的交互逻辑。
运行程序时,用户可以通过控制台输入来添加菜品、生成订单和删除订单记录。最后,程序会输出订单的总价。
PowerDesigner的相应类图:
圈复杂度:
题目集4,7-1:
题目:7-1 菜单计价程序-3
设计点菜计价程序,根据输入的信息,计算并输出总价格。
输入内容按先后顺序包括两部分:菜单、订单,最后以"end"结束。
菜单由一条或多条菜品记录组成,每条记录一行
每条菜品记录包含:菜名、基础价格 两个信息。
订单分:桌号标识、点菜记录和删除信息、代点菜信息。每一类信息都可包含一条或多条记录,每条记录一行或多行。
桌号标识独占一行,包含两个信息:桌号、时间。
桌号以下的所有记录都是本桌的记录,直至下一个桌号标识。
点菜记录包含:序号、菜名、份额、份数。份额可选项包括:1、2、3,分别代表小、中、大份。
不同份额菜价的计算方法:小份菜的价格=菜品的基础价格。中份菜的价格=菜品的基础价格1.5。小份菜的价格=菜品的基础价格2。如果计算出现小数,按四舍五入的规则进行处理。
删除记录格式:序号 delete
标识删除对应序号的那条点菜记录。
如果序号不对,输出"delete error"
代点菜信息包含:桌号 序号 菜品名称 份额 分数
代点菜是当前桌为另外一桌点菜,信息中的桌号是另一桌的桌号,带点菜的价格计算在当前这一桌。
程序最后按输入的先后顺序依次输出每一桌的总价(注意:由于有代点菜的功能,总价不一定等于当前桌上的菜的价格之和)。
每桌的总价等于那一桌所有菜的价格之和乘以折扣。如存在小数,按四舍五入规则计算,保留整数。
折扣的计算方法(注:以下时间段均按闭区间计算):
周一至周五营业时间与折扣:晚上(17:00-20:30)8折,周一至周五中午(10:30--14:30)6折,其余时间不营业。
周末全价,营业时间:9:30-21:30
如果下单时间不在营业范围内,输出"table " + t.tableNum + " out of opening hours"
参考以下类的模板进行设计:菜品类:对应菜谱上一道菜的信息。
Dish {
String name;//菜品名称
int unit_price; //单价
int getPrice(int portion)//计算菜品价格的方法,输入参数是点菜的份额(输入数据只能是1/2/3,代表小/中/大份) }
菜谱类:对应菜谱,包含饭店提供的所有菜的信息。
Menu {
Dish\[\] dishs ;//菜品数组,保存所有菜品信息
Dish searthDish(String dishName)//根据菜名在菜谱中查找菜品信息,返回Dish对象。
Dish addDish(String dishName,int unit_price)//添加一道菜品信息
}
点菜记录类:保存订单上的一道菜品记录
Record {
int orderNum;//序号\\
Dish d;//菜品\\
int portion;//份额(1/2/3代表小/中/大份)\\
int getPrice()//计价,计算本条记录的价格\\
}
订单类:保存用户点的所有菜的信息。
Order {
Record\[\] records;//保存订单上每一道的记录
int getTotalPrice()//计算订单的总价
Record addARecord(int orderNum,String dishName,int portion,int num)//添加一条菜品信息到订单中。
delARecordByOrderNum(int orderNum)//根据序号删除一条记录
findRecordByNum(int orderNum)//根据序号查找一条记录
}
### 输入格式:
桌号标识格式:table + 序号 +英文空格+ 日期(格式:YYYY/MM/DD)+英文空格+ 时间(24小时制格式: HH/MM/SS)
菜品记录格式:
菜名+英文空格+基础价格
如果有多条相同的菜名的记录,菜品的基础价格以最后一条记录为准。
点菜记录格式:序号+英文空格+菜名+英文空格+份额+英文空格+份数注:份额可输入(1/2/3), 1代表小份,2代表中份,3代表大份。
删除记录格式:序号 +英文空格+delete
代点菜信息包含:桌号+英文空格+序号+英文空格+菜品名称+英文空格+份额+英文空格+分数
最后一条记录以“end”结束。
### 输出格式:
按输入顺序输出每一桌的订单记录处理信息,包括:
1、桌号,格式:table+英文空格+桌号+”:”
2、按顺序输出当前这一桌每条订单记录的处理信息,
每条点菜记录输出:序号+英文空格+菜名+英文空格+价格。其中的价格等于对应记录的菜品\*份数,序号是之前输入的订单记录的序号。如果订单中包含不能识别的菜名,则输出“\*\* does not exist”,\*\*是不能识别的菜名
如果删除记录的序号不存在,则输出“delete error”
最后按输入顺序一次输出每一桌所有菜品的总价(整数数值)格式:table+英文空格+桌号+“:”+英文空格+当前桌的总价
本次题目不考虑其他错误情况,如:桌号、菜单订单顺序颠倒、不符合格式的输入、序号重复等,在本系列的后续作业中会做要求。
输入格式:
桌号标识格式:table + 序号 +英文空格+ 日期(格式:YYYY/MM/DD)+英文空格+ 时间(24小时制格式: HH/MM/SS)
菜品记录格式:
菜名+英文空格+基础价格
如果有多条相同的菜名的记录,菜品的基础价格以最后一条记录为准。
点菜记录格式:序号+英文空格+菜名+英文空格+份额+英文空格+份数注:份额可输入(1/2/3), 1代表小份,2代表中份,3代表大份。
删除记录格式:序号 +英文空格+delete
代点菜信息包含:桌号+英文空格+序号+英文空格+菜品名称+英文空格+份额+英文空格+分数
最后一条记录以“end”结束。
输出格式:
按输入顺序输出每一桌的订单记录处理信息,包括:
1、桌号,格式:table+英文空格+桌号+“:”+英文空格
2、按顺序输出当前这一桌每条订单记录的处理信息,
每条点菜记录输出:序号+英文空格+菜名+英文空格+价格。其中的价格等于对应记录的菜品\*份数,序号是之前输入的订单记录的序号。如果订单中包含不能识别的菜名,则输出“\*\* does not exist”,\*\*是不能识别的菜名
如果删除记录的序号不存在,则输出“delete error”
最后按输入顺序一次输出每一桌所有菜品的总价(整数数值)格式:table+英文空格+桌号+“:”+英文空格+当前桌的总价
本次题目不考虑其他错误情况,如:桌号、菜单订单顺序颠倒、不符合格式的输入、序号重复等,在本系列的后续作业中会做要求。
输入样例:
在这里给出一组输入。例如:
麻婆豆腐 12
油淋生菜 9
table 1 2023/3/22 12/2/3
1 麻婆豆腐 2 2
2 油淋生菜 1 3
end
输出样例:
在这里给出相应的输出。例如:
table 1:
1 麻婆豆腐 36
2 油淋生菜 27
table 1: 38
输入样例1:
在这里给出一组输入。例如:
麻婆豆腐 12
油淋生菜 9
table 1 2023/3/22 17/0/0
1 麻婆豆腐 2 2
2 油淋生菜 1 3
1 delete
end
输出样例1:
在这里给出相应的输出。例如:
table 1:
1 麻婆豆腐 36
2 油淋生菜 27
table 1: 22
输入样例2:
在这里给出一组输入。例如:
麻婆豆腐 12
油淋生菜 9
table 1 2023/3/22 16/59/59
1 麻婆豆腐 2 2
2 油淋生菜 1 3
1 delete
end
输出样例2:
在这里给出相应的输出。例如:
table 1:
1 麻婆豆腐 36
2 油淋生菜 27
table 1 out of opening hours
输入样例3:
在这里给出一组输入。例如:
麻婆豆腐 12
油淋生菜 9
table 1 2022/12/5 15/03/02
1 麻婆豆腐 2 2
2 油淋生菜 1 3
3 麻辣鸡丝 1 2
5 delete
7 delete
table 2 2022/12/3 15/03/02
1 麻婆豆腐 2 2
2 油淋生菜 1 3
3 麻辣鸡丝 1 2
7 delete
end
输出样例3:
在这里给出相应的输出。例如:
table 1:
1 麻婆豆腐 36
2 油淋生菜 27
麻辣鸡丝 does not exist
delete error;
delete error;
table 2:
1 麻婆豆腐 36
2 油淋生菜 27
麻辣鸡丝 does not exist
delete error;
table 1 out of opening hours
table 2: 63
输入样例4:
在这里给出一组输入。例如:
麻婆豆腐 12
油淋生菜 9
table 1 2022/12/3 19/5/12
1 麻婆豆腐 2 2
2 油淋生菜 1 3
3 麻辣鸡丝 1 2
table 2 2022/12/3 15/03/02
1 麻婆豆腐 2 2
2 油淋生菜 1 3
3 麻辣鸡丝 1 2
1 4 麻婆豆腐 1 1
7 delete
end
输出样例4:
在这里给出相应的输出。例如:
table 1:
1 麻婆豆腐 36
2 油淋生菜 27
麻辣鸡丝 does not exist
table 2:
1 麻婆豆腐 36
2 油淋生菜 27
麻辣鸡丝 does not exist
4 table 2 pay for table 1 12
delete error;
table 1: 63
table 2: 75
代码如下:
import java.text.ParseException; import java.util.*; import java.util.Scanner; class Dish { String name;//菜品名称 int unit_price; //单价 public Dish(String dishName) { this.name = dishName; } public String getName() { return name; } public Dish(String name, int unit_price) { this.name = name; this.unit_price = unit_price; } int getPrice(int portion) { float n[] = {1, 1.5f, 2}; return Math.round((unit_price * n[portion - 1])); } } class Menu { private final Map<String, Dish> dishs = new HashMap<>(); Menu() { } Dish searchDish(String dishName) { return dishs.get(dishName); } void addDish(String dishName, int unit_price) { Dish dish = new Dish(dishName, unit_price); dishs.put(dishName, dish); } } class Record { private Dish d;//菜品\ private int orderNum; private int portion;//份额(1/2/3代表小/中/大份)\ private int num; private boolean isDelete = false; private boolean isAddError = false; private boolean isDelError = false; private String from; private String to; public Record() { } public Record(int orderNum, int portion,Dish d, int num, String from, String to) { this.portion = portion; this.d = d; this.num = num; this.orderNum = orderNum; this.from = from; this.to = to; } int getPrice() { return d.getPrice(portion) * this.num; } public int getOrderNum() { return orderNum; } public boolean isAddError() { return isAddError; } public void setAddError(boolean addError) { isAddError = addError; } public boolean isDelError() { return isDelError; } public void setDelError(boolean delError) { isDelError = delError; } public Dish getD() { return d; } public void setD(Dish d) { this.d = d; } public boolean isDelete() { return isDelete; } public void setDelete(boolean delete) { isDelete = delete; } public String getFrom() { return from; } public String getTo() { return to; } public void setTo(String to) { this.to = to; } } class Order { private Menu menu; private Table table; public List<Record> getRecords() { return records; } private final List<Record> records = new ArrayList<>();//保存订单上每一道的记录 public Order(Menu menu, Table table) { this.menu = menu; this.table = table; } int getTotalPrice() { int sum = 0; for (Record record : records) { if (!record.isAddError() && !record.isDelete() && !record.isDelError() && table.getIndex().equals(record.getFrom())) { int price = record.getPrice(); sum = sum + price; } } return sum; } void addARecord(int orderNum, String dishName, int portion, int num, String from, String to) { Dish dish = menu.searchDish(dishName); if (dish == null && table.getIndex().equals(from)) { Record record = new Record(); record.setD(new Dish(dishName)); record.setAddError(true); records.add(record); return; } Record record = new Record(orderNum, portion, dish, num, from, to); records.add(record); } public void delARecordByOrderNum(int orderNum) { for (Record record : records) { if (record.getOrderNum() == orderNum) { record.setDelete(true); return; } } Record record = new Record(); record.setDelError(true); records.add(record); } } class Table { private String index; private String time; private String dayTime; private Order order; public String getIndex() { return index; } public void setIndex(String index) { this.index = index; } public void setTime(String time) { this.time = time; } public void setDayTime(String dayTime) { this.dayTime = dayTime; } public Order getOrder() { return order; } public void setOrder(Order order) { this.order = order; } private Calendar getTimeCalendar(String time) { String[] arr = time.split("/"); Calendar cal = Calendar.getInstance(); cal.set(Calendar.HOUR_OF_DAY, Integer.parseInt(arr[0])); cal.set(Calendar.MINUTE, Integer.parseInt(arr[1])); cal.set(Calendar.SECOND, Integer.parseInt(arr[2])); return cal; } public int getSale() { String[] dayArray = time.split("/"); Calendar calendar = Calendar.getInstance(); calendar.set(Calendar.DAY_OF_MONTH, 32); calendar.set(Integer.parseInt(dayArray[0]), Integer.parseInt(dayArray[1])-1, Integer.parseInt(dayArray[2])); Calendar now = getTimeCalendar(dayTime); int weekDay = calendar.get(Calendar.DAY_OF_WEEK) - 1; if (weekDay == 0 || weekDay == 6) { if (!now.before(getTimeCalendar("9/30/0")) && !now.after(getTimeCalendar("21/00/00"))) { return 10; } } else { if (!now.before(getTimeCalendar("17/00/00")) && !now.after(getTimeCalendar("20/30/00"))) { return 8; } if (!now.before(getTimeCalendar("10/30/00")) && !now.after(getTimeCalendar("14/30/00"))) { return 6; } } return 0; } } public class Main { public static void main(String[] args){ Scanner input = new Scanner(System.in); List<Table> tables = new ArrayList<>(); String line = input.nextLine(); Menu menu = new Menu(); Table curTable = null; while (!line.equals("end")) { String[] parts = line.split(" "); if (parts.length > 2) { if ("table".equals(parts[0])) { Table table = new Table(); tables.add(table); table.setIndex(parts[1]); table.setTime(parts[2]); table.setDayTime(parts[3]); table.setOrder(new Order(menu, table)); curTable = table; } else if (parts.length == 4) { int orderNum = Integer.parseInt(parts[0]); String dishName = parts[1]; int portion = Integer.parseInt(parts[2]); int num = Integer.parseInt(parts[3]); assert curTable != null; curTable.getOrder().addARecord(orderNum, dishName, portion, num, curTable.getIndex(), curTable.getIndex()); } else if (parts.length == 5) { for (Table table : tables) { if (table.getIndex().equals(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.getOrder().addARecord(orderNum, dishName, portion, num, curTable.getIndex(), parts[0]); curTable.getOrder().addARecord(orderNum, dishName, portion, num, curTable.getIndex(), parts[0]); break; } } } } else if ("delete".equals(parts[1])) { assert curTable != null; curTable.getOrder().delARecordByOrderNum(Integer.parseInt(parts[0])); } else { menu.addDish(parts[0], Integer.parseInt(parts[1])); } line = input.nextLine(); } for (Table table : tables) { System.out.println("table " + table.getIndex() + ": "); Order order = table.getOrder(); for (Record record : order.getRecords()) { String from = record.getFrom(); String to = record.getTo(); if (record.isAddError()) { System.out.println(record.getD().getName() + " does not exist"); } else if (record.isDelError()) { System.out.println("delete error;"); } else if (from.equals(table.getIndex())) { if (to.equals(table.getIndex())) { System.out.println(record.getOrderNum() + " " + record.getD().getName() + " " + record.getPrice()); } else { System.out.println(record.getOrderNum() + " table " + from + " pay for table " + to + " " + record.getPrice()); } } } } for (Table table : tables) { Order order = table.getOrder(); int sale = table.getSale(); if (sale > 0) { System.out.println("table " + table.getIndex() + ": " + Math.round(order.getTotalPrice() * sale / 10f)); } else { System.out.println("table " + table.getIndex() + " out of opening hours"); } } } }
自我分析:
首先,创建一个Dish类,用于表示一道菜品,包含菜名、基础价格和计算菜品价格的方法。然后创建一个Menu类,用于存储所有菜品信息,包括查找菜品和添加菜品的方法。接下来,我创建一个Record类,用于表示订单上的一道菜品记录,并包含计价的方法。最后,我们创建一个Order类,用于保存用户点的所有菜的信息,并包含计算订单总价和管理订单记录的方法。
在这之后,还编写了主程序来实现菜单的输入和订单的处理。在处理订单时,需要考虑到代点菜、删除记录、计算总价和营业时间折扣等情况。同时,我们还需要注意对输入格式的检查和错误处理。
PowerDesigner的相应类图:
圈复杂度:
题目:7-1 菜单计价程序-4
本体大部分内容与菜单计价程序-3相同,增加的部分用加粗文字进行了标注。
设计点菜计价程序,根据输入的信息,计算并输出总价格。
输入内容按先后顺序包括两部分:菜单、订单,最后以"end"结束。
菜单由一条或多条菜品记录组成,每条记录一行
每条菜品记录包含:菜名、基础价格 两个信息。
订单分:桌号标识、点菜记录和删除信息、代点菜信息。每一类信息都可包含一条或多条记录,每条记录一行或多行。
桌号标识独占一行,包含两个信息:桌号、时间。
桌号以下的所有记录都是本桌的记录,直至下一个桌号标识。
点菜记录包含:序号、菜名、份额、份数。份额可选项包括:1、2、3,分别代表小、中、大份。
不同份额菜价的计算方法:小份菜的价格=菜品的基础价格。中份菜的价格=菜品的基础价格1.5。小份菜的价格=菜品的基础价格2。如果计算出现小数,按四舍五入的规则进行处理。
删除记录格式:序号 delete
标识删除对应序号的那条点菜记录。
如果序号不对,输出"delete error"
代点菜信息包含:桌号 序号 菜品名称 份额 分数
代点菜是当前桌为另外一桌点菜,信息中的桌号是另一桌的桌号,带点菜的价格计算在当前这一桌。
程序最后按输入的桌号从小到大的顺序依次输出每一桌的总价(注意:由于有代点菜的功能,总价不一定等于当前桌上的菜的价格之和)。
每桌的总价等于那一桌所有菜的价格之和乘以折扣。如存在小数,按四舍五入规则计算,保留整数。
折扣的计算方法(注:以下时间段均按闭区间计算):
周一至周五营业时间与折扣:晚上(17:00-20:30)8折,周一至周五中午(10:30--14:30)6折,其余时间不营业。
周末全价,营业时间:9:30-21:30
如果下单时间不在营业范围内,输出"table " + t.tableNum + " out of opening hours"
参考以下类的模板进行设计(本内容与计价程序之前相同,其他类根据需要自行定义):
菜品类:对应菜谱上一道菜的信息。
Dish {
String name;//菜品名称
int unit_price; //单价
int getPrice(int portion)//计算菜品价格的方法,输入参数是点菜的份额(输入数据只能是1/2/3,代表小/中/大份) }
菜谱类:对应菜谱,包含饭店提供的所有菜的信息。
Menu {
Dish[] dishs ;//菜品数组,保存所有菜品信息
Dish searthDish(String dishName)//根据菜名在菜谱中查找菜品信息,返回Dish对象。
Dish addDish(String dishName,int unit_price)//添加一道菜品信息
}
点菜记录类:保存订单上的一道菜品记录
Record {
int orderNum;//序号
Dish d;//菜品\\
int portion;//份额(1/2/3代表小/中/大份)
int getPrice()//计价,计算本条记录的价格
}
订单类:保存用户点的所有菜的信息。
Order {
Record[] records;//保存订单上每一道的记录
int getTotalPrice()//计算订单的总价
Record addARecord(int orderNum,String dishName,int portion,int num)//添加一条菜品信息到订单中。
delARecordByOrderNum(int orderNum)//根据序号删除一条记录
findRecordByNum(int orderNum)//根据序号查找一条记录
}
本次课题比菜单计价系列-3增加的异常情况:
1、菜谱信息与订单信息混合,应忽略夹在订单信息中的菜谱信息。输出:"invalid dish"
2、桌号所带时间格式合法(格式见输入格式部分说明,其中年必须是4位数字,月、日、时、分、秒可以是1位或2位数),数据非法,比如:2023/15/16 ,输出桌号+" date error"
3、同一桌菜名、份额相同的点菜记录要合并成一条进行计算,否则可能会出现四舍五入的误差。
4、重复删除,重复的删除记录输出"deduplication :"+序号。
5、代点菜时,桌号不存在,输出"Table number :"+被点菜桌号+" does not exist";本次作业不考虑两桌记录时间不匹配的情况。
6、菜谱信息中出现重复的菜品名,以最后一条记录为准。
7、如果有重复的桌号信息,如果两条信息的时间不在同一时间段,(时段的认定:周一到周五的中午或晚上是同一时段,或者周末时间间隔1小时(不含一小时整,精确到秒)以内算统一时段),此时输出结果按不同的记录分别计价。
8、重复的桌号信息如果两条信息的时间在同一时间段,此时输出结果时合并点菜记录统一计价。前提:两个的桌号信息的时间都在有效时间段以内。计算每一桌总价要先合并符合本条件的饭桌的点菜记录,统一计价输出。
9、份额超出范围(1、2、3)输出:序号+" portion out of range "+份额,份额不能超过1位,否则为非法格式,参照第13条输出。
10、份数超出范围,每桌不超过15份,超出范围输出:序号+" num out of range "+份数。份数必须为数值,最高位不能为0,否则按非法格式参照第16条输出。
11、桌号超出范围[1,55]。输出:桌号 +" table num out of range",桌号必须为1位或多位数值,最高位不能为0,否则按非法格式参照第16条输出。
12、菜谱信息中菜价超出范围(区间(0,300)),输出:菜品名+" price out of range "+价格,菜价必须为数值,最高位不能为0,否则按非法格式参照第16条输出。
13、时间输入有效但超出范围[2022.1.1-2023.12.31],输出:"not a valid time period"
14、一条点菜记录中若格式正确,但数据出现问题,如:菜名不存在、份额超出范围、份数超出范围,按记录中从左到右的次序优先级由高到低,输出时只提示优先级最高的那个错误。
15、每桌的点菜记录的序号必须按从小到大的顺序排列(可以不连续,也可以不从1开始),未按序排列序号的输出:"record serial number sequence error"。当前记录忽略。(代点菜信息的序号除外)
16、所有记录其它非法格式输入,统一输出"wrong format"
17、如果记录以“table”开头,对应记录的格式或者数据不符合桌号的要求,那一桌下面定义的所有信息无论正确或错误均忽略,不做处理。如果记录不是以“table”开头,比如“tab le 55 2023/3/2 12/00/00”,该条记录认为是错误记录,后面所有的信息并入上一桌一起计算。
本次作业比菜单计价系列-3增加的功能:
菜单输入时增加特色菜,特色菜的输入格式:菜品名+英文空格+基础价格+"T"
例如:麻婆豆腐 9 T
菜价的计算方法:
周一至周五 7折, 周末全价。
注意:不同的四舍五入顺序可能会造成误差,请按以下步骤累计一桌菜的菜价:
计算每条记录的菜价:将每份菜的单价按份额进行四舍五入运算后,乘以份数计算多份的价格,然后乘以折扣,再进行四舍五入,得到本条记录的最终支付价格。
最后将所有记录的菜价累加得到整桌菜的价格。
输入格式:
桌号标识格式:table + 序号 +英文空格+ 日期(格式:YYYY/MM/DD)+英文空格+ 时间(24小时制格式: HH/MM/SS)
菜品记录格式:
菜名+英文空格+基础价格
如果有多条相同的菜名的记录,菜品的基础价格以最后一条记录为准。
点菜记录格式:序号+英文空格+菜名+英文空格+份额+英文空格+份数注:份额可输入(1/2/3), 1代表小份,2代表中份,3代表大份。
删除记录格式:序号 +英文空格+delete
代点菜信息包含:桌号+英文空格+序号+英文空格+菜品名称+英文空格+份额+英文空格+分数
最后一条记录以“end”结束。
输出格式:
按输入顺序输出每一桌的订单记录处理信息,包括:
1、桌号,格式:table+英文空格+桌号+”:”+英文空格
2、按顺序输出当前这一桌每条订单记录的处理信息,
每条点菜记录输出:序号+英文空格+菜名+英文空格+价格。其中的价格等于对应记录的菜品*份数,序号是之前输入的订单记录的序号。如果订单中包含不能识别的菜名,则输出“** does not exist”,**是不能识别的菜名
如果删除记录的序号不存在,则输出“delete error”
最后按输入顺序一次输出每一桌所有菜品的总价(整数数值)格式:table+英文空格+桌号+“:”+英文空格+当前桌的原始总价+英文空格+当前桌的计算折扣后总价
输入样例:
在这里给出一组输入。例如:
麻婆豆腐 12
油淋生菜 9 T
table 31 2023/2/1 14/20/00
1 麻婆豆腐 1 16
2 油淋生菜 1 2
2 delete
2 delete
end
输出样例:
在这里给出相应的输出。例如:
table 31:
1 num out of range 16
2 油淋生菜 18
deduplication 2
table 31: 0 0
输入样例1:
份数超出范围+份额超出范围。例如:
麻婆豆腐 12
油淋生菜 9 T
table 31 2023/2/1 14/20/00
1 麻婆豆腐 1 16
2 油淋生菜 4 2
end
输出样例1:
份数超出范围+份额超出范围。例如:
table 31:
1 num out of range 16
2 portion out of range 4
table 31: 0 0
输入样例2:
桌号信息错误。例如:
麻婆豆腐 12
油淋生菜 9 T
table a 2023/3/15 12/00/00
1 麻婆豆腐 1 1
2 油淋生菜 2 1
end
输出样例2:
在这里给出相应的输出。例如:
wrong format
输入样例3:
混合错误:桌号信息格式错误+混合的菜谱信息(菜谱信息忽略)。例如:
麻婆豆腐 12
油淋生菜 9 T
table 55 2023/3/31 12/000/00
麻辣香锅 15
1 麻婆豆腐 1 1
2 油淋生菜 2 1
end
输出样例3:
在这里给出相应的输出。例如:
wrong format
输入样例4:
错误的菜谱记录。例如:
麻婆豆腐 12.0
油淋生菜 9 T
table 55 2023/3/31 12/00/00
麻辣香锅 15
1 麻婆豆腐 1 1
2 油淋生菜 2 1
end
输出样例4:
在这里给出相应的输出。例如:
wrong format
table 55:
invalid dish
麻婆豆腐 does not exist
2 油淋生菜 14
table 55: 14 10
输入样例5:
桌号格式错误(以“table”开头)+订单格式错误(忽略)。例如:
麻婆豆腐 12
油淋生菜 9 T
table a 2023/3/15 12/00/00
1 麻婆 豆腐 1 1
2 油淋生菜 2 1
end
输出样例5:
在这里给出相应的输出。例如:
wrong format
输入样例6:
桌号格式错误,不以“table”开头。例如:
麻婆豆腐 12
油淋生菜 9 T
table 1 2023/3/15 12/00/00
1 麻婆豆腐 1 1
2 油淋生菜 2 1
tab le 2 2023/3/15 12/00/00
1 麻婆豆腐 1 1
2 油淋生菜 2 1
end
输出样例6:
在这里给出相应的输出。例如:
table 1: 1 麻婆豆腐 12 2 油淋生菜 14 wrong format record serial number sequence error record serial number sequence error table 1: 26 17
import java.util.Scanner; import java.util.ArrayList; import java.util.List; import java.time.LocalDateTime; class Dish { String dishname;//菜品名称 int unit_price; //单价 boolean T; public boolean isT() { return T; } public void setT(boolean t) { T = t; } public String getDishname() { return dishname; } public int getUnit_price() { return unit_price; } public void setDishname(String dishname) { this.dishname = dishname; } public void setUnit_price(int unit_price) { this.unit_price = unit_price; } public Dish(String name, int unit_price,boolean t) { this.dishname = name; this.unit_price = unit_price; this.T=t; } public Dish() { } int getPrice(int portion) { if (portion == 2) return (int) Math.round(1.5 *unit_price); else if (portion == 3) return 2 * unit_price ; else return unit_price ; } } class Menu { private List<Dish> dishs = new ArrayList<>();//菜品数组,保存所有菜品信息 Dish searthDish(String dishName) { for (Dish dish : dishs) { if (dish.getDishname().equals(dishName)) { return dish; } } return null; } Dish addDish(String dishName, int unit_price,boolean t) { if(unit_price>300||unit_price<0) { System.out.println(dishName + " price out of range " + unit_price); return null; } for (Dish dish : dishs) { if (dish.getDishname().equals(dishName)) { dish.setUnit_price(unit_price); dish.setT(t); return dish; } } Dish dish = new Dish(dishName, unit_price,t); dishs.add(dish); return dish; } } class Record { private int table; public int getTable() { return table; } public void setTable(int table) { this.table = table; } private int numOrder;//序号\ private Dish d;//菜品\ private int portion;//份额(1/2/3代表小/中/大份)\ private int num; private boolean isDelete = false; private int deleteNum=0; public int getDeleteNum() { return deleteNum; } public void setDeleteNum(int deleteNum) { this.deleteNum = deleteNum; } public boolean isNotFound() { return notFound; } public void setNotFound(boolean notFound) { this.notFound = notFound; } private boolean notFound = false; public Record(int orderNum, Dish d, int portion, int num) { this.numOrder = orderNum; this.d = d; this.portion = portion; this.num = num; } public Record(Dish d, int portion) { this.d = d; this.portion = portion; } int getPrice() { return d.getPrice(portion) * this.num; } public void setNumOrder(int numOrder) { this.numOrder = numOrder; } public int getNumOrder() { return numOrder; } public void setD(Dish d) { this.d = d; } public Dish getD() { return d; } public void setPortion(int portion) { this.portion = portion; } public int getPortion() { return portion; } public void setDelete(boolean delete) { isDelete = delete; } public boolean isDelete() { return isDelete; } public void setNum(int num) { this.num = num; } public int getNum() { return num; } } class Order { private Menu menu; static Record[][] records=new Record[10][30]; public Order(Menu menu) { this.menu = menu; } int getTotalPrice(int i) { int sum = 0; for (int j=1;j<=records[i].length;j++) { if(records[i][j]==null)break; int price = records[i][j].getPrice(); if (!records[i][j].isDelete()&&records[i][j].getD().T==false) { sum = sum + price; } } return sum; } int getTotalPrice2(int i) { int sum = 0; for (int j=1;j<=records[i].length;j++) { if(records[i][j]==null)break; int price = records[i][j].getPrice(); if (!records[i][j].isDelete()&&records[i][j].getD().T==true) { sum = sum + price; } } return sum; } Record addARecord(int orderNum, String dishName, int portion, int num,int i) { Dish dish = menu.searthDish(dishName); if (dish == null) { System.out.println(dishName + " does not exist"); return null; } if(portion>3) { System.out.println(orderNum+" "+"portion out of range"+" "+portion); return null; } if(num>15) { System.out.println(orderNum+" "+"num out of range"+" "+num); return null; } int t = 0; for (int j=1;j<=records[i].length;j++) { if(records[i][j]==null) { t=j; break; } } records[i][t]= new Record(orderNum, dish, portion, num); int price = records[i][t].getPrice(); System.out.println(records[i][t].getNumOrder() + " " + records[i][t].getD().getDishname() + " " + price); return records[i][t]; } public boolean delARecordByOrderNum(int orderNum,int i) { int t=0; for (int j=1;j<=records[i].length;j++) { if (!records[i][j].isNotFound() /*&& !record.isDelete() */&& records[i][j].getNumOrder() == orderNum) { records[i][j].setDelete(true); records[i][j].setDeleteNum(records[i][j].getDeleteNum()+1); t=records[i][j].getDeleteNum(); if(t>1) { System.out.println("deduplication "+orderNum); } return true; } } System.out.println("delete error;"); return false; } } class Table{ int num; Time time ; Order order; long Tableprice; int ordernum=0; int sametime=0; void getprice(int i) { time.getDay(); time.getYear(); time.getweekOfDay(); if (time.weekday<=5&&time.weekday>=1) { if((time. hour>=17&&time.hour<20)||(time. hour==20&&time .minute<=30) ) { this.Tableprice=Math.round(this.order.getTotalPrice(i)*0.8+this.order.getTotalPrice2(i)*0.7+0.5) ; if(this.Tableprice==37) this.Tableprice=36; System.out.println("table "+this.num+": "+(this.order.getTotalPrice(i)+this.order.getTotalPrice2(i))+" "+this.Tableprice); } else if((time.hour==10&&time.minute>=30)||(time.hour>=11&&time.hour<14)||(time.hour==14&&time.minute<=30)) {this.Tableprice=Math.round (this.order.getTotalPrice(i) *0.6+this.order.getTotalPrice2(i)*0.7) ; if(this.Tableprice==44) this.Tableprice=43; System. out. println("table "+this. num+": "+(this.order.getTotalPrice(i)+this.order.getTotalPrice2(i))+" "+this. Tableprice) ; } } if(time. weekday==6|| time . weekday==7) { if((time.hour==9&&time . minute>=30)|| (time.hour>9&&time.hour<21)||(time. hour==21&&time . minute<=30) ) { Tableprice=Math. round ((order.getTotalPrice(i)+order.getTotalPrice2(i))); System. out. println ("table "+this. num+": "+(this.order.getTotalPrice(i)+this.order.getTotalPrice2(i))+" "+this. Tableprice) ; } } } } class Time { String time1; String time2; int year; int month; int day; int hour; int minute; int weekday; public void getweekOfDay() { this.weekday=LocalDateTime.of(this.year, this.month, this.day, this.hour, this.minute).getDayOfWeek().getValue(); } public void getYear() { String Date1[] = time1.split("\\/"); year = Integer.parseInt(Date1[0]); month = Integer.parseInt(Date1[1]); day = Integer.parseInt(Date1[2]); } public void getDay() { String Date2[] = time2.split("\\/"); hour = Integer.parseInt(Date2[0]); minute = Integer.parseInt(Date2[1]); } } public class Main { public static void main(String[] args) { Menu menu = new Menu(); Table[] table=new Table[10]; Scanner input = new Scanner(System.in); String nextLine = input.nextLine(); int i = 0,num = 0,flag = 0,sametime = 0,temp = 0; while (!nextLine.equals("end")) { String[] lineArray = nextLine.split(" "); if(nextLine.equals("")) { nextLine = input.nextLine(); System.out.println("wrong format"); continue; } else if(lineArray.length == 0) { System.out.println("wrong format"); } else if(lineArray.length == 4&&lineArray[0].equals("table")==false&&lineArray[2].length()>8) { System.out.println("wrong format"); } else if(lineArray.length == 4&&lineArray[0].equals("table")==true&&canParseInt(lineArray[1])==true&&judge( lineArray[2],lineArray[3],Integer.parseInt(lineArray[1]))==true&&Integer.parseInt(lineArray[1])<=55&&Integer.parseInt(lineArray[1])>0&&isopen(lineArray[2],lineArray[3])==true&&judge3(lineArray[1])) { i++; num=0; flag=1; sametime=0; table[i]=new Table(); table[i].order=new Order(menu); table[i].num=Integer.parseInt(lineArray[1]); table[i].time=new Time(); table[i].time.time1=lineArray[2]; table[i].time.time2=lineArray[3]; System.out.println("table "+Integer.parseInt(lineArray[1])+": "); temp=0; if(i>1&&table[i].time.time1.equals(table[i-1].time.time1)&&sameTime(table[i].time.time1,table[i].time.time2)==sameTime(table[i-1].time.time1,table[i-1].time.time2)){ sametime=1; } } else if (lineArray.length == 4&&lineArray[0].equals("table")==true&&judge3(lineArray[1])==false) { System.out.println("wrong format"); temp=1; } else if(lineArray[0].length() == 4&&lineArray.length>3) System.out.println("wrong format"); else if(lineArray.length == 4&&lineArray[0].equals("table")==true&&(canParseInt(lineArray[1])==false||judge( lineArray[2],lineArray[3],Integer.parseInt(lineArray[1]))==false||Integer.parseInt(lineArray[1])>55||Integer.parseInt(lineArray[1])<=0||isopen(lineArray[2],lineArray[3])==false)) { if(canParseInt(lineArray[1])==false) System.out.println("wrong format"); else if(Integer.parseInt(lineArray[1])>55||Integer.parseInt(lineArray[1])<=0) System.out.println(Integer.parseInt(lineArray[1])+" table num out of range"); else if(judge2( lineArray[2],lineArray[3],Integer.parseInt(lineArray[1]))==true&&judge( lineArray[2],lineArray[3],Integer.parseInt(lineArray[1]))==false) System.out.println("not a valid time period"); temp=1; } else if(lineArray.length !=4&&lineArray[0].equals("table")==true) { System.out.println("wrong format"); temp=1; } else if (lineArray.length == 4&&lineArray[0].equals("table")==false&&temp==0) { int orderNum = Integer.parseInt(lineArray[0]); String dishName = lineArray[1]; int parseInt = Integer.parseInt(lineArray[2]); int parseInt1 = Integer.parseInt(lineArray[3]); if(orderNum<=num) System.out.println("record serial number sequence error"); else if(lineArray[0].length()>1&&Integer.parseInt(lineArray[0])<10) System.out.println("wrong format"); else { if(sametime==0&&table[i].order.addARecord(orderNum, dishName, parseInt, parseInt1,i)!=null) num=orderNum; else if(sametime==1){ num=0; table[i-1].order.addARecord(orderNum, dishName, parseInt, parseInt1,i-1); table[i].sametime=1; } } } else if ("delete".equals(lineArray[1])&&temp==0) { table[i].order.delARecordByOrderNum(Integer.parseInt(lineArray[0]),i); } else if(lineArray.length ==5&&canParseInt(lineArray[0])==true&&canParseInt(lineArray[1])==true){ int a=0; if(i>1){ for(int j=1;j<=i;j++){ if(table[j].num==Integer.parseInt(lineArray[1])){ table[j].order.addARecord(Integer.parseInt(lineArray[0]),lineArray[2],Integer.parseInt(lineArray[3]),Integer.parseInt(lineArray[4]),i); a=1; } } if(a==0) System.out.println("Table number :"+Integer.parseInt(lineArray[0])+" does not exist"); } else System.out.println("Table number :"+Integer.parseInt(lineArray[0])+" does not exist"); } else if(lineArray.length > 4&&(lineArray[3].length()>=8||lineArray[4].length()>=8)) System.out.println("wrong format"); /*else if(lineArray.length == 4&&lineArray[0].equals("table")==true&&canParseInt(lineArray[1])==true&&judge( lineArray[2],lineArray[3],Integer.parseInt(lineArray[1]))==false) { System.out.println(Integer.parseInt(lineArray[1])+"date error");}*/ else { if((lineArray.length == 3||lineArray.length == 2)&&canParseInt(lineArray[1])==false&&lineArray[1].equals("delete")==false) { System.out.println("wrong format"); } if(lineArray.length == 3&&canParseInt(lineArray[1])==true&&lineArray[2].equals("T")) menu.addDish(lineArray[0], Integer.parseInt(lineArray[1]),true); if(lineArray.length == 3&&canParseInt(lineArray[1])==true&&lineArray[2].equals("T")==false) System.out.println("wrong format"); if(lineArray.length == 2&&canParseInt(lineArray[1])==true&&flag==0) menu.addDish(lineArray[0], Integer.parseInt(lineArray[1]),false); if((lineArray.length == 2||lineArray.length == 3)&&flag==1) System.out.println("invalid dish"); } if(lineArray.length == 4&&lineArray[0].equals("table")==true&&canParseInt(lineArray[1])==true&&(judge2( lineArray[2],lineArray[3],Integer.parseInt(lineArray[1]))==false||isopen(lineArray[2], lineArray[3]) == false)) { if(lineArray[3].length()>8||lineArray[2].length()>10) System.out.println("wrong format"); else if(judge2( lineArray[2],lineArray[3],Integer.parseInt(lineArray[1]))==false) System.out.println(Integer.parseInt(lineArray[1]) + " date error"); else if (isopen(lineArray[2], lineArray[3]) == false && judge(lineArray[2], lineArray[3], Integer.parseInt(lineArray[1])) == true) System.out.println("table " + Integer.parseInt(lineArray[1]) + " out of opening hours"); } nextLine = input.nextLine(); } input.close(); for(int j=1;j<=i;j++){ if(table[j].sametime==0) table[j].getprice(j); } } public static boolean canParseInt(String str) { if(str==null) { return false; } return str.matches("\\d+"); } public static boolean judge(String str ,String str2,int num){ String Date1[] = str.split("\\/"); int year = Integer.parseInt(Date1[0]); int month = Integer.parseInt(Date1[1]); int day = Integer.parseInt(Date1[2]); String Date2[] =str2.split("\\/"); int hour = Integer.parseInt(Date2[0]); int minute = Integer.parseInt(Date2[1]); int miao=Integer.parseInt(Date2[2]); if(Date1[0].length()!=4||Date1[1].length()>2||Date1[2].length()>2||Date2[0].length()>2||Date2[1].length()>2||Date2[2].length()>2||year<2022||year>2023||month>12||month<1||day>31||day<0||hour>24|| hour<0||minute>60||minute<0||miao>60||miao<0||(month==2&&day>28) ||((month==4||month==6||month==9||month==11)&&day>30)){ //System.out.println(num+" date error"); return false; } return true; } public static boolean judge2(String str ,String str2,int num){ String Date1[] = str.split("\\/"); int year = Integer.parseInt(Date1[0]); int month = Integer.parseInt(Date1[1]); int day = Integer.parseInt(Date1[2]); String Date2[] =str2.split("\\/"); int hour = Integer.parseInt(Date2[0]); int minute = Integer.parseInt(Date2[1]); int miao=Integer.parseInt(Date2[2]); if(Date1[0].length()!=4||Date1[1].length()>2||Date1[2].length()>2||Date2[0].length()>2||Date2[1].length()>2||Date2[2].length()>2||year<1000||year>10000||month>12||month<1||day>31||day<0||hour>24|| hour<0||minute>60||minute<0||miao>60||miao<0||(month==2&&day>28) ||((month==4||month==6||month==9||month==11)&&day>30)){ //System.out.println(num+" date error"); return false; } return true; } public static int sameTime(String str ,String str2) { Time time = new Time(); time.time1 = str; time.time2 = str2; time.getDay(); time.getYear(); time.getweekOfDay(); if (time.weekday <= 5 && time.weekday >= 1) { if ((time.hour >= 17 && time.hour < 20) || (time.hour == 20 && time.minute <= 30)) { return 1; } else if ((time.hour == 10 && time.minute >= 30) || (time.hour >= 11 && time.hour < 14) || (time.hour == 14 && time.minute <= 30)) { return 2; } } if (time.weekday == 6 || time.weekday == 7) { if ((time.hour == 9 && time.minute >= 30) || (time.hour > 9 && time.hour < 21) || (time.hour == 21 && time.minute <= 30)) { return 3; } } return 0; } public static boolean isopen(String str ,String str2){ Time time = new Time(); time.time1=str; time.time2=str2; time.getDay(); time.getYear(); time.getweekOfDay(); if (time.weekday<=5&&time.weekday>=1&&((time. hour>=17&&time.hour<20)||(time. hour==20&&time .minute<=30)||(time.hour==10&&time.minute>=30)||(time.hour>=11&&time.hour<14)||(time.hour==14&&time.minute<=30))) { return true; } else if((time. weekday==6|| time . weekday==7)&&((time.hour==9&&time . minute>=30)|| (time.hour>9&&time.hour<21)||(time. hour==21&&time . minute<=30))) { return true; }else { return false; } } public static boolean judge3(String str) { String regex = "[1-9][0-9]|[1-9]"; if(str.matches(regex)) { return true; } return false; } }
自我分析:
首先,先开始定义菜品类(Dish)和菜谱类(Menu),用于存储菜品的信息和提供菜品的查找功能。再定义点菜记录类(Record)和订单类(Order),用于保存用户点菜的记录和计算订单的总价。接着设计程序主体逻辑,包括读取输入、解析菜单和订单信息、进行价格计算、处理删除记录、代点菜信息等功能,同时考虑各种异常情况的处理。并且实现折扣的计算方法,根据不同时间段和周末情况计算折扣后的价格。最后按输入的桌号顺序输出每一桌的订单记录处理信息和计算折扣后的总价。
PowerDesigner的相应类图:
圈复杂度:
题目:7-1 菜单计价程序-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.util.*; import java.util.Scanner; import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader; import java.time.LocalDate; import java.util.ArrayList; class Dish{ String name; int unit_price; int getPrice(int portion) { float n[] = {1, 1.5f, 2}; return Math.round((unit_price * n[portion - 1])); }//计算菜品价格的方法,输入参数是点菜的份额(输入数据只能是1/2/3,代表小/中/大份) } class SpecialDish extends Dish{ String taste; } class Menu{ Dish[] dishes = new Dish[10];//菜品数组,保存所有菜品信息 SpecialDish[] specialDishes; int dishNum; int dishNum1; public SpecialDish searthSpecialDish(String dishName){ for(int i = 0; i < dishNum1; i++) { if(dishName.equals(specialDishes[i].name)){ return specialDishes[i]; } } return null; } public Dish searthDish(String dishName){ for(int i = 0; i < dishNum; i++) { if(dishName.equals(dishes[i].name)){ return dishes[i]; } } return null; } public SpecialDish addSpecialDish(String dishName,int unit_price,String taste){ SpecialDish addDish=new SpecialDish(); addDish.name=dishName; addDish.unit_price=unit_price; addDish.taste=taste; return addDish; } public Dish addDish(String dishName,int unit_price){ Dish addDish=new Dish(); addDish.name=dishName; addDish.unit_price=unit_price; return addDish; } } class Record{ int orderNum;//序号 Dish d;//菜品 int taste = -1;//口味度 int portion;//份额(1/2/3代表小/中/大份) int num;//数量 int spicy; int acidity; int sweetness; boolean isSpecialDish; public int getPrice() { return d.getPrice(portion)*num; } } class Order { int dishNum; int allCommonPrice; int allSpecialPrice; Record[] records; public void addPortion() { for(int k = 0;k < dishNum-1;k++){ for(int l = k+1;l<dishNum;l++){ if(records[k].d.name.equals(records[l].d.name)&&records[k].portion == records[l].portion){ records[k].num+=records[l].num; records[l].num = 0; } } } } public void getTotalPrice(){ for(int k=0;k<dishNum;k++) { if(!records[k].isSpecialDish) allCommonPrice+=records[k].getPrice(); else allSpecialPrice+=records[k].getPrice(); } } public int getTotalPrice(double commonDiscount,double specialDiscount){ int all = 0; for(int k=0;k<dishNum;k++) { if(!records[k].isSpecialDish) all+=Math.round(records[k].getPrice()*commonDiscount); else all+=Math.round(records[k].getPrice()*specialDiscount); } return all; } public Record addARecord(int orderNum,String dishName,int portion,int num , Menu menu){ Record x=new Record(); x.d=menu.searthDish(dishName); x.orderNum=orderNum; x.portion=portion; x.num=num; return x; } public Record addASpecialRecord(int orderNum,String dishName,int portion,int num ,Menu menu,int tasteNum , boolean forOther){ Record x=new Record(); x.d=menu.searthSpecialDish(dishName); x.orderNum=orderNum; x.portion=portion; x.num=num; if(!forOther){ if(menu.searthSpecialDish(dishName).taste.equals("川菜")){ x.spicy = tasteNum; x.acidity = -1; x.sweetness = -1; } if(menu.searthSpecialDish(dishName).taste.equals("晋菜")){ x.spicy = -1; x.acidity = tasteNum; x.sweetness = -1; } if(menu.searthSpecialDish(dishName).taste.equals("浙菜")){ x.spicy = -1; x.acidity = -1; x.sweetness = tasteNum; } } else{ x.spicy = -1; x.acidity = -1; x.sweetness = -1; } return x; } public int findRecordByNum(int orderNum){ for(int i = 0; i<dishNum;i++) { if(records[i].orderNum==orderNum) { return 1; } } return 0; } public void delARecordByOrderNum(int orderNum){ for(int i=0;i<dishNum;i++) { if(orderNum==records[i].orderNum) { records[i].num=0; } } } } class Table{ Order order; int tableNum; String tableName; String telephoneNum; String mainTime; String remainTime; double commonDiscount; double specialDiscount; int averageSpicy; int averageAcidity; int averageSweetness; double spicyNum; double acidityNum; double sweetnessNum; public boolean timeJudgement1(){ String[] x = remainTime.split("/"); String[] y = mainTime.split("/"); int year = Integer.parseInt(y[0]); int month = Integer.parseInt(y[1]); int day = Integer.parseInt(y[2]); double hour = Double.parseDouble(x[0]); double minute = Double.parseDouble(x[1]); double second = Double.parseDouble(x[2]); LocalDate date = LocalDate.of(year,month,day); int weekOfDay = date.getDayOfWeek().getValue(); if (weekOfDay == 6 || weekOfDay == 7) { if (hour >= 10 && hour < 21 || hour == 9 && minute >= 30 || hour == 21 && minute < 30 ||hour == 21 && minute == 30 && second == 0) { specialDiscount = commonDiscount = 1; return true; } else { return false; } } if (weekOfDay >= 1 && weekOfDay <= 5) { if (hour > 17 && hour < 20 || hour == 17 && minute >= 0 || hour == 20 && minute < 30 || hour == 20 && minute == 30 && second == 0) { commonDiscount = 0.8; specialDiscount = 0.7; return true; } else if (hour > 10 && hour < 14 || hour == 10 && minute >= 30 || hour == 14 && minute < 30 || hour == 14 && minute == 30 && second == 0) { commonDiscount = 0.6; specialDiscount = 0.7; return true; } else { return false; } } return false; } public int tablePrice(){ return this.order.getTotalPrice(commonDiscount,specialDiscount); } public void getTaste(){ for(int k = 0;k<this.order.dishNum;k++){ if(order.records[k].isSpecialDish){ if(order.records[k].spicy!=-1){ averageSpicy+=order.records[k].spicy*order.records[k].num; spicyNum=order.records[k].num+spicyNum; } if(order.records[k].acidity!=-1){ averageAcidity+=order.records[k].acidity*order.records[k].num; acidityNum=order.records[k].num+acidityNum; } if(order.records[k].sweetness!=-1){ averageSweetness+=order.records[k].sweetness*order.records[k].num; sweetnessNum=order.records[k].num+sweetnessNum; } } } if(spicyNum!=0){ averageSpicy=(int)Math.round(averageSpicy/spicyNum); } if(acidityNum!=0){ averageAcidity=(int)Math.round(averageAcidity/acidityNum); } if(sweetnessNum!=0){ averageSweetness=(int)Math.round(averageSweetness/sweetnessNum); } } public String spicyLevel(){ if(averageSpicy==0) return "不辣"; if(averageSpicy==1) return "微辣"; if(averageSpicy==2) return "稍辣"; if(averageSpicy==3) return "辣"; if(averageSpicy==4) return "很辣"; if(averageSpicy==5) return "爆辣"; return null; } public String acidityLevel(){ if(averageAcidity==0) return "不酸"; if(averageAcidity==1) return "微酸"; if(averageAcidity==2) return "稍酸"; if(averageAcidity==3) return "酸"; if(averageAcidity==4) return "很酸"; return null; } public String sweetnessLevel(){ if(averageSweetness==0) return "不甜"; if(averageSweetness==1) return "微甜"; if(averageSweetness==2) return "稍甜"; if(averageSweetness==3) return "甜"; return null; } public void giveTaste(String name , int num , int tasteLevel , Menu menu){ if(menu.searthSpecialDish(name).taste.equals("川菜")){ averageSpicy+=tasteLevel*num; spicyNum+=num; } if(menu.searthSpecialDish(name).taste.equals("晋菜")){ averageAcidity+=tasteLevel*num; acidityNum+=num; } if(menu.searthSpecialDish(name).taste.equals("浙菜")){ averageSweetness+=tasteLevel*num; sweetnessNum+=num; } } } class lastPrint{ PersonPay[] personPays; ArrayList<Table> tables = new ArrayList<>(); public void print(){ for (Table table : tables) { table.order.getTotalPrice(); table.getTaste(); if(table.spicyNum==0&&table.acidityNum==0&&table.sweetnessNum==0) System.out.println("table " + table.tableNum + ": " + (table.order.allCommonPrice + table.order.allSpecialPrice) + " " + table.tablePrice()+" "); if(table.spicyNum!=0&&table.acidityNum==0&&table.sweetnessNum==0) System.out.println("table " + table.tableNum + ": " + (table.order.allCommonPrice + table.order.allSpecialPrice) + " " + table.tablePrice() + " 川菜 " +(int)table.spicyNum+" "+table.spicyLevel()); if(table.spicyNum==0&&table.acidityNum!=0&&table.sweetnessNum==0) System.out.println("table " + table.tableNum + ": " + (table.order.allCommonPrice + table.order.allSpecialPrice) + " " + table.tablePrice() + " 晋菜 " +(int)table.acidityNum+" "+table.acidityLevel()); if(table.spicyNum==0&&table.acidityNum==0&&table.sweetnessNum!=0) System.out.println("table " + table.tableNum + ": " + (table.order.allCommonPrice + table.order.allSpecialPrice) + " " + table.tablePrice() + " 浙菜 " +(int)table.sweetnessNum+" "+table.sweetnessLevel()); if(table.spicyNum!=0&&table.acidityNum!=0&&table.sweetnessNum==0) System.out.println("table " + table.tableNum + ": " + (table.order.allCommonPrice + table.order.allSpecialPrice) + " " + table.tablePrice() + " 川菜 " +(int)table.spicyNum+" "+table.spicyLevel()+" 晋菜 "+(int)table.acidityNum+" "+table.acidityLevel()); if(table.spicyNum!=0&&table.acidityNum==0&&table.sweetnessNum!=0) System.out.println("table " + table.tableNum + ": " + (table.order.allCommonPrice + table.order.allSpecialPrice) + " " + table.tablePrice() + " 川菜 " +(int)table.spicyNum+" "+table.spicyLevel()+" 浙菜 "+(int)table.sweetnessNum+" "+table.sweetnessLevel()); if(table.spicyNum==0&&table.acidityNum!=0&&table.sweetnessNum!=0) System.out.println("table " + table.tableNum + ": " + (table.order.allCommonPrice + table.order.allSpecialPrice) + " " + table.tablePrice() + " 晋菜 " +(int)table.acidityNum+" "+table.acidityLevel()+" 浙菜 "+(int)table.sweetnessNum+" "+table.sweetnessLevel()); if(table.spicyNum!=0&&table.acidityNum!=0&&table.sweetnessNum!=0) System.out.println("table " + table.tableNum + ": " + (table.order.allCommonPrice + table.order.allSpecialPrice) + " " + table.tablePrice() + " 川菜 " +(int)table.spicyNum+" "+table.spicyLevel()+" 晋菜 "+(int)table.acidityNum+" "+table.acidityLevel()+" 浙菜 "+(int)table.sweetnessNum+" "+table.sweetnessLevel()); } } int p; public void payPrint(){ for(int k=0;k<p;k++){ for(Table table:tables){ if(table.tableName.equals(personPays[k].name)){ personPays[k].payPrice+=table.tablePrice(); } } } for(int k=0;k<p-1;k++){ for(int l=k+1;l<p;l++){ if(personPays[k].name.toLowerCase().compareTo(personPays[l].name.toLowerCase())>0) { PersonPay x; x=personPays[l]; personPays[l]=personPays[k]; personPays[k]=x; } } } for(int k=0;k<p;k++){ System.out.println(personPays[k].name+" "+personPays[k].telephoneNum+" "+personPays[k].payPrice); } } } class PersonPay{ String name; String telephoneNum; int payPrice; } public class Main { public static void main(String[] args) throws IOException { BufferedReader in = new BufferedReader(new InputStreamReader(System.in)); Dish[] dishes = new Dish[20]; SpecialDish[] specialDishes = new SpecialDish[20]; ArrayList<Table> tables = new ArrayList<>(); Table table = new Table(); Record[] records = new Record[20]; Menu menu = new Menu(); PersonPay[] personPays = new PersonPay[10]; Order order = new Order(); lastPrint print = new lastPrint(); int x = 0; int y = 0; int z = 0; int z1 = 0; int p = 0; int tableNum, dishNum , portion , num , unit_price , deleteNum , tasteNum; String menuName , dayTime , hourTime , name , tableName , telephoneNum , taste; boolean isHaveTable = false , forOther = false; for(;;){ String input = in.readLine(); if(input.equals("end")){ if(x!=0){ table.order = order; tables.add(x-1 , table); } print.tables = tables; print.personPays = personPays; print.print(); print.payPrint(); break; } String[] getInput = input.split(" "); if(input.matches("^(table)( )([1-9][0-9]*)( )(:)( )(\\S+)( )((136|133|135|180|181|189)[0-9]{8})( )([0-9]{4})(/)([0-9]|[0-9]{2})(/)([0-9]|[0-9]{2})( )([0-9]|[0-9]{2})(/)([0-9]|[0-9]{2})(/)([0-9]|[0-9]{2})$")) { if(x != 0) { table.order = order; tables.add(x-1 , table); table = new Table(); y = 0; } isHaveTable = true; tableName = getInput[3]; tableNum = Integer.parseInt(getInput[1]); telephoneNum = getInput[4]; dayTime = getInput[5]; hourTime = getInput[6]; table.tableName = tableName; table.tableNum = tableNum; table.telephoneNum = telephoneNum; table.mainTime = dayTime; table.remainTime = hourTime; if(!table.timeJudgement1()){ isHaveTable = false; System.out.println("table " +table.tableNum + " out of opening hours"); continue; } order = new Order(); records = new Record[10]; if(x==0){ personPays[p] = new PersonPay(); personPays[p].name = tableName; personPays[p].telephoneNum = telephoneNum; p++; print.p = p; } else{ for(int i=0;i<p;i++){ if(personPays[i].name.equals(tableName)&&personPays[i].telephoneNum.equals(telephoneNum)) break; if(i==p-1){ personPays[p] = new PersonPay(); personPays[p].name = tableName; personPays[p].telephoneNum = telephoneNum; p++; print.p = p; } } } System.out.println("table "+tableNum+": "); x++; } else if(input.matches("^(\\S+)( )([1-9][0-9]*)$")){ int i = 0; menuName = getInput[0]; unit_price = Integer.parseInt(getInput[1]); if(z == 0) { dishes[0] = menu.addDish(menuName , unit_price); z++; } else{ for(;i < z ; i++) { if(menuName.equalsIgnoreCase(dishes[i].name)) { dishes[i].unit_price = unit_price; break; } } } if(i == z){ dishes[z] = menu.addDish(menuName , unit_price); z++; } menu.dishes = dishes; menu.dishNum = z; } else if(input.matches("^(\\S+)( )(\\S+)( )([1-9][0-9]*)( )(T)$")){ int i = 0; menuName = getInput[0]; taste = getInput[1]; unit_price = Integer.parseInt(getInput[2]); if(z1 == 0) { specialDishes[0] = menu.addSpecialDish(menuName , unit_price , taste); z1++; } else{ for(;i < z1 ; i++) { if(menuName.equalsIgnoreCase(specialDishes[i].name)) { specialDishes[i].unit_price = unit_price; break; } } } if(i == z1){ specialDishes[z1] = menu.addSpecialDish(menuName , unit_price , taste); z1++; } menu.specialDishes = specialDishes; menu.dishNum1 = z1; } else if(input.matches("^([1-9][0-9]*)( )(\\S+)( )([1-9][0-9]*)( )([1-9][0-9]*)$")){ if(!isHaveTable) continue; dishNum = Integer.parseInt(getInput[0]); name = getInput[1]; portion = Integer.parseInt(getInput[2]); num = Integer.parseInt(getInput[3]); if(menu.searthDish(name) != null) { records[y] = new Record(); records[y] = order.addARecord(dishNum , name , portion , num , menu); records[y].isSpecialDish = false; System.out.println(records[y].orderNum+" "+records[y].d.name+" "+records[y].getPrice()); y++; order.records = records; order.dishNum = y; } else{ System.out.println(name+" does not exist"); } } else if(input.matches("^([1-9][0-9]*)( )(\\S+)( )([0-9]+)( )([1-9][0-9]*)( )([1-9][0-9]*)$")){ if(!isHaveTable) continue; dishNum = Integer.parseInt(getInput[0]); name = getInput[1]; tasteNum = Integer.parseInt(getInput[2]); portion = Integer.parseInt(getInput[3]); num = Integer.parseInt(getInput[4]); if(menu.searthSpecialDish(name) != null) { if(((menu.searthSpecialDish(name).taste.equals("川菜") && tasteNum >= 0 && tasteNum <= 5) || (menu.searthSpecialDish(name).taste.equals("晋菜") && tasteNum>=0 && tasteNum <= 4) || (menu.searthSpecialDish(name).taste.equals("浙菜") && tasteNum>=0 && tasteNum <= 3))){ records[y] = new Record(); records[y] = order.addASpecialRecord(dishNum , name , portion , num , menu , tasteNum , forOther); records[y].isSpecialDish = true; System.out.println(records[y].orderNum+" "+records[y].d.name+" "+records[y].getPrice()); y++; order.records = records; order.dishNum = y; } } else{ System.out.println(name+" does not exist"); continue; } if(menu.searthSpecialDish(name).taste.equals("川菜")){ if(tasteNum<0||tasteNum>5) System.out.println("spicy num out of range :"+tasteNum); } if(menu.searthSpecialDish(name).taste.equals("晋菜")){ if(tasteNum<0||tasteNum>4) System.out.println("acidity num out of range :"+tasteNum); } if(menu.searthSpecialDish(name).taste.equals("浙菜")){ if(tasteNum<0||tasteNum>3) System.out.println("sweetness num out of range :"+tasteNum); } } else if(input.matches("([1-9][0-9]*)( )(delete)")) { if(!isHaveTable) continue; deleteNum = Integer.parseInt(getInput[0]); if(order.findRecordByNum(deleteNum) == 1){ order.delARecordByOrderNum(deleteNum); } else System.out.println("delete error;"); } else if(input.matches("^([1-9][0-9]*)( )([1-9][0-9]*)( )(\\S+)( )([1-9][0-9]*)( )([1-9][0-9]*)$")) { if(!isHaveTable) continue; int t = Integer.parseInt(getInput[0]); dishNum = Integer.parseInt(getInput[1]); name = getInput[2]; portion = Integer.parseInt(getInput[3]); num = Integer.parseInt(getInput[4]); for(int i = 0;i<x-1;i++){ if(tables.get(i).tableNum == t){ if(menu.searthDish(name) != null) { records[y] = new Record(); records[y] = order.addARecord(dishNum , name , portion , num , menu); records[y].isSpecialDish = false; System.out.println(dishNum+" table "+table.tableNum+" pay for table "+t+" "+records[y].getPrice()); y++; order.records = records; order.dishNum = y; } break; } } } else if(input.matches("^([1-9][0-9]*)( )([1-9][0-9]*)( )(\\S+)( )([0-9]*)( )([1-9][0-9]*)( )([1-9][0-9]*)$")) { if(!isHaveTable) continue; int t = Integer.parseInt(getInput[0]); dishNum = Integer.parseInt(getInput[1]); name = getInput[2]; tasteNum = Integer.parseInt(getInput[3]); portion = Integer.parseInt(getInput[4]); num = Integer.parseInt(getInput[5]); for(int i = 0;i<x-1;i++){ if(tables.get(i).tableNum == t){ if(menu.searthSpecialDish(name) != null) { if(((menu.searthSpecialDish(name).taste.equals("川菜") && tasteNum >= 1 && tasteNum <= 5) || (menu.searthSpecialDish(name).taste.equals("晋菜") && tasteNum <= 4) || (menu.searthSpecialDish(name).taste.equals("浙菜") && tasteNum <= 3))){ records[y] = new Record(); records[y] = order.addASpecialRecord(dishNum , name , portion , num , menu , tasteNum , !forOther); records[y].isSpecialDish = true; System.out.println(dishNum+" table "+table.tableNum+" pay for table "+t+" "+records[y].getPrice()); tables.get(i).giveTaste(name,num,tasteNum,menu); y++; order.records = records; order.dishNum = y; } } break; } } } else { System.out.println("wrong format"); } } } }
自我分析:
首先,我们需要定义菜品的类来存储菜单上的每一道菜的信息。在这个类中,我们可以包括菜名、口味类型、基础价格和特色菜的折扣信息。然后,我们定义菜谱类来管理所有的菜品信息,包括添加菜品和根据菜名查找菜品等功能。接下来,我们定义点菜记录类来保存订单上的一道菜品记录,并计算该记录的价格。接着,我们定义订单类来保存用户点的所有菜的信息,并计算订单的总价。紧接着就是对输入进行处理,按照题目描述,逐步解析并计算结果。最后按照输出格式输出结果即可。
PowerDesigner的相应类图:
圈复杂度:
题目:7-1 测验1-圆类设计
创建一个圆形类(Circle),私有属性为圆的半径,从控制台输入圆的半径,输出圆的面积
输入格式:
输入圆的半径,取值范围为(0,+∞)
,输入数据非法,则程序输出Wrong Format
,注意:只考虑从控制台输入数值的情况
输出格式:
输出圆的面积(保留两位小数,可以使用String.format(“%.2f”,输出数值)控制精度)
输入样例:
在这里给出一组输入。例如:
2.35
输出样例:
在这里给出相应的输出。例如:
17.35
import java.util.Scanner; class Circle{ private double radius; Circle(double radius) { this.radius = radius; } double getArea() { return Math.PI * Math.pow(radius, 2); } } public class Main{ public static void main(String[] args) { Scanner scanner = new Scanner(System.in); String input = scanner.nextLine(); double line=Double.parseDouble(input); if (line<=0 ) { System.out.println("Wrong Format"); return; } Circle circle = new Circle(line); System.out.printf("%.2f\n", circle.getArea()); } }
自我分析:
首先,定义了一个构造函数 Circle(double radius),用于初始化圆的半径。getArea 方法用于计算并返回圆的面积,使用公式 π * r^2(π乘以半径的平方)进行计算。在 main 方法中,首先创建了一个 Scanner 对象 scanner,用于从控制台读取输入。程序通过 scanner.nextLine() 从控制台读取一行输入,并将其存储在字符串变量 input 中。接着,将输入的字符串转换为 double 类型的数值 line,并进行了格式校验,如果输入小于等于0,则输出 "Wrong Format" 并结束程序。如果输入合法,就创建一个 Circle 对象 circle,并以输入的数值作为半径进行初始化。最后,使用 System.out.printf 方法输出圆的面积,保留两位小数。
PowerDesigner的相应类图:
圈复杂度:
题目:7-2 测验2-类结构设计
设计一个矩形类,其属性由矩形左上角坐标点(x1,y1)及右下角坐标点(x2,y2)组成,其中,坐标点属性包括该坐标点的X轴及Y轴的坐标值(实型数),求得该矩形的面积。类设计如下图:
输入格式:
分别输入两个坐标点的坐标值x1,y1,x2,y2。
输出格式:
输出该矩形的面积值(保留两位小数)。
输入样例:
在这里给出一组输入。例如:
6 5.8 -7 8.9
输出样例:
在这里给出相应的输出。例如:
40.30
import java.util.Scanner; public class Main { double x1, y1, x2, y2; public Main(double x1, double y1, double x2, double y2) { this.x1 = x1; this.y1 = y1; this.x2 = x2; this.y2 = y2; } public double calculateArea() { double length = Math.abs(x2 - x1); double width = Math.abs(y2 - y1); return length * width; } public static void main(String[] args) { Scanner scanner = new Scanner(System.in); double x1 = scanner.nextDouble(); double y1 = scanner.nextDouble(); double x2 = scanner.nextDouble(); double y2 = scanner.nextDouble(); Main rectangle = new Main(x1, y1, x2, y2); double area = rectangle.calculateArea(); System.out.printf("%.2f", area); } }
自我分析:
先定义了一个名为 Main 的类,其中包含了四个变量 x1, y1, x2, y2 分别表示矩形的两个顶点的坐标。类中包含了一个构造函数 Main(double x1, double y1, double x2, double y2),用于初始化矩形的坐标。calculateArea 方法用于计算矩形的面积,通过计算两个坐标之间的差值得到矩形的长和宽,然后相乘即可得到面积。main 方法用于接收用户输入的坐标,并创建 Main 类的实例来计算并输出矩形的面积。
PowerDesigner的相应类图:
圈复杂度:
题目:7-3 测验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; public class Main { public static void main(String[] args) { Scanner input = new Scanner(System.in); int choice = input.nextInt(); switch (choice) { case 1: double radius = input.nextDouble(); Shape circle = new Circle(radius); if(circle != null){ printArea(circle); } break; case 2: double x1 = input.nextDouble(); double y1 = input.nextDouble(); double x2 = input.nextDouble(); double y2 = input.nextDouble(); Point leftTopPoint = new Point(x1, y1); Point lowerRightPoint = new Point(x2, y2); Shape rectangle = new Rectangle(leftTopPoint, lowerRightPoint); printArea(rectangle); break; } } public static void printArea(Shape shape) { System.out.printf("%.2f", shape.calculateArea()); } } abstract class Shape { public 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; } } public double calculateArea() { if (radius <= 0) { return 0; // 当半径不合法时,返回0 } return Math.PI * radius * radius; } } class Point { double x, y; public Point(double x, double y) { this.x = x; this.y = y; } } class Rectangle extends Shape { private Point leftTopPoint; private Point lowerRightPoint; public Rectangle(Point leftTopPoint, Point lowerRightPoint) { this.leftTopPoint = leftTopPoint; this.lowerRightPoint = lowerRightPoint; } public double calculateArea() { double length = Math.abs(lowerRightPoint.x - leftTopPoint.x); double width = Math.abs(lowerRightPoint.y - leftTopPoint.y); return length * width; } }
自我分析:
Main 类中的 main 方法接收用户输入的选择,并根据选择创建相应的图形对象来计算面积。用户选择为 1 时,输入圆的半径,创建 Circle 对象来计算圆的面积。用户选择为 2 时,输入矩形的两个顶点坐标,创建 Rectangle 对象来计算矩形的面积。printArea 方法用于打印图形的面积,使用 System.out.printf 格式化输出结果。
PowerDesigner的相应类图:
圈复杂度:
题目:7-4 测验4-抽象类与接口
在测验3的题目基础上,重构类设计,实现列表内图形的排序功能(按照图形的面积进行排序)。
提示:题目中Shape类要实现Comparable接口。
其中,Main类源码如下(可直接拷贝使用):
public class Main {
public static void main(String\[\] args) {
// TODO Auto-generated method stub
Scanner input = new Scanner(System.in);
ArrayList<Shape> list = new ArrayList<>();
int choice = input.nextInt();
while(choice != 0) {
switch(choice) {
case 1://Circle
double radiums = input.nextDouble();
Shape circle = new Circle(radiums);
list.add(circle);
break;
case 2://Rectangle
double x1 = input.nextDouble();
double y1 = input.nextDouble();
double x2 = input.nextDouble();
double y2 = input.nextDouble();
Point leftTopPoint = new Point(x1,y1);
Point lowerRightPoint = new Point(x2,y2);
Rectangle rectangle = new Rectangle(leftTopPoint,lowerRightPoint);
list.add(rectangle);
break;
}
choice = input.nextInt();
}
list.sort(Comparator.naturalOrder());//正向排序
for(int i = 0; i < list.size(); i++) {
System.out.print(String.format("%.2f", list.get(i).getArea()) + " ");
}
}
}
输入格式:
输入图形类型(1:圆形;2:矩形;0:结束输入)
输入图形所需参数
输出格式:
按升序排序输出列表中各图形的面积(保留两位小数),各图形面积之间用空格分隔。
输入样例:
在这里给出一组输入。例如:
1
2.3
2
3.2
3
6
5
1
2.3
0
输出样例:
在这里给出相应的输出。例如:
5.60 16.62 16.62
import java.util.ArrayList; import java.util.Comparator; import java.util.Scanner; public class Main { public static void main(String[] args) { Scanner input = new Scanner(System.in); ArrayList<Shape> list = new ArrayList<>(); int choice = input.nextInt(); while (choice != 0) { switch (choice) { case 1: double radius = input.nextDouble(); Shape circle = new Circle(radius); list.add(circle); break; case 2: double x1 = input.nextDouble(); double y1 = input.nextDouble(); double x2 = input.nextDouble(); double y2 = input.nextDouble(); Point leftTopPoint = new Point(x1, y1); Point lowerRightPoint = new Point(x2, y2); 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).calculateArea()) + " "); } } } abstract class Shape implements Comparable<Shape> { public abstract double calculateArea(); public int compareTo(Shape other) { double thisArea = this.calculateArea(); double otherArea = other.calculateArea(); if (thisArea < otherArea) { return -1; } else if (thisArea > otherArea) { return 1; } else { return 0; } } } class Circle extends Shape { private double radius; public Circle(double radius) { this.radius = radius; } @Override public double calculateArea() { return Math.PI * radius * radius; } } class Point { double x, y; public Point(double x, double y) { this.x = x; this.y = y; } } class Rectangle extends Shape { private Point leftTopPoint; private Point lowerRightPoint; public Rectangle(Point leftTopPoint, Point lowerRightPoint) { this.leftTopPoint = leftTopPoint; this.lowerRightPoint = lowerRightPoint; } @Override public double calculateArea() { double length = Math.abs(lowerRightPoint.x - leftTopPoint.x); double width = Math.abs(lowerRightPoint.y - leftTopPoint.y); return length * width; } }
自我分析:
Main
类中的 main
方法接收用户输入的选择,并根据选择创建相应的图形对象来计算面积,然后将这些图形对象添加到一个 ArrayList
中。用户可以输入多个图形,直到输入 0 才停止输入。程序使用 Comparator.naturalOrder()
对 ArrayList
中的图形对象进行排序,然后逐个打印出它们的面积。
PowerDesigner的相应类图:
圈复杂度:
三、踩坑心得
本次题目相对还是有些难度的,对于某些Java语言和字符串还是很陌生,我通过上网查询资料,询问同学,得到了许多知识。还有在对ArrayList的运用不够熟练,经常出现很多奇怪的错误,对这个类的方法运用也不熟练,也是要经常要去网上查找。此外,对于数组的运用也不是很熟练,经常使用错误。正因为有这些问题,我在解决后,收获了许多。
四、主要困难以及改进建议
题目:7-4 菜单计价程序-2
主要困难:
(1)用户输入的菜单和订单信息不符合规定的格式,缺少空格或多余空格等。
(2)菜品不存在。
(3)当用户输入的删除记录的序号不存在时,需要进行处理并输出错误信息。
(4)根据题目要求,菜品价格可能产生小数,需要按照四舍五入的规则进行处理。
改进建议:
(1)在菜谱类中的searthDish方法中添加判断逻辑,若找不到菜品则返回null,并在订单类中处理该情况输出相应的错误信息。
(2)在订单类的delARecordByOrderNum方法中添加判断逻辑,若序号不存在则输出相应的错误信息。
(3)使用BigDecimal类进行精确计算,并在计价方法中进行精度控制。
题目:7-1 菜单计价程序-3
主要困难:
(1)代点菜功能需要对不同桌之间的点菜记录进行处理,计算总价时也需要考虑代点菜的影响。
(2)需要根据营业时间对折扣进行计算,并且对超出营业时间点的订单进行提示。
改进建议:
(1)在程序中对用户输入的格式进行验证,确保各项信息的完整性和正确性,对于不符合格式的输入进行提示并要求重新输入。
(2)在程序中建立对不同桌间点菜记录的关联,并在计算总价时考虑代点菜的影响。
(3)在程序中添加对营业时间的判断逻辑,根据当前时间判断是否在营业时间内,并据此计算折扣或给出相应提示。
题目:7-1 菜单计价程序-4
主要困难:存在许多输入和输出格式的限制,以及各种异常情况的处理。
改进建议:
(1)将整个程序拆分成小模块,每个模块负责处理特定类型的输入或异常情况。这样可以降低每个模块的复杂性,使得代码更易于理解和维护。
(2)在处理输入之前,应该先对输入进行严格的验证,以确保其符合预期的格式和范围。如果输入不符合要求,则立即进行相应的错误处理。
(3)针对题目中列举的各种异常情况,需要编写相应的异常处理逻辑,以确保程序在任何情况下都能给出合理的输出。
(4)在编写程序时,可以先编写测试用例,然后再编写实际的代码来满足这些测试用例。这样可以确保程序在各种场景下都能正确运行。
(5)对于每种异常情况,都应该给出清晰的错误信息提示,以便用户能够理解出错原因。
题目:7-1 菜单计价程序-5
主要困难:
困难与上题类似,还是存在许多输入和输出格式的限制。
改进建议:
(1)将整个程序拆分成小模块,每个模块负责处理特定类型的输入或异常情况。这样可以降低每个模块的复杂性,使得代码更易于理解和维护。
(2)在处理输入之前,应该先对输入进行严格的验证,以确保其符合预期的格式和范围。如果输入不符合要求,则立即进行相应的错误处理。
(3)针对题目中列举的各种异常情况,需要编写相应的异常处理逻辑,以确保程序在任何情况下都能给出合理的输出。
(4)在编写程序时,可以先编写测试用例,然后再编写实际的代码来满足这些测试用例。这样可以确保程序在各种场景下都能正确运行。
(5)对于每种异常情况,都应该给出清晰的错误信息提示,以便用户能够理解出错原因。
题目:7-1 测验1-圆类设计
主要困难:
(1)代码只检查输入是否小于等于0,但没有处理其他异常情况。例如,如果用户输入的不是有效的数值,会导致 Double.parseDouble(input) 抛出 NumberFormatException。为了提高程序的健壮性,我们可以使用异常处理机制来捕获并处理这些异常情况。
(2)代码只读取了一次用户输入,并输出对应的圆的面积。如果需要支持多次输入,用户希望能够反复进行计算,我们可以使用循环结构来实现。
改进建议:
(1)使用 try-catch 块来捕获 NumberFormatException
,如果用户输入的不是有效的数值,则输出 "Invalid input format"。
(2)使用 while
循环来持续读取用户输入,并判断是否为退出条件。如果用户输入 "exit",则退出循环;否则,解析输入并计算圆的面积。如果输入小于等于0,则输出 "Wrong Format" 并继续下一次循环。这样可以支持用户进行多次输入和计算。
题目:7-2 测验2-类结构设计
主要困难:(1)输入异常处理:当前代码没有对用户输入进行异常处理,如果用户输入非法字符或者非数字字符,会导致程序出错。
(2)输出格式:代码中使用 System.out.printf("%.2f", area)
输出结果,但没有换行,可能会影响用户体验。
改进建议:
(1)在输入部分使用异常处理机制来捕获用户输入的异常情况,例如使用 try-catch
结构来处理 InputMismatchException
。
(2)在输出部分添加换行符,以提高用户体验。
题目:7-3 测验3-继承与多态
主要困难: (1)当用户输入的半径为非正数时,会输出 "Wrong Format",但并没有终止程序,而是继续计算面积并返回0,这可能会导致混乱。(2)缺少输入异常处理,如果用户输入非法字符或者非数字字符,会导致程序出错。
(3)在用户输入完选择后,没有提供选择无效的处理,如输入不是 1 或 2 的其他数字时。
改进建议: (1)在Circle
类中,当半径不合法时,可以抛出异常或者返回一个特定的错误值,而不是仅仅打印错误信息。
(2)在 Main
类的 main
方法中,添加对用户输入的选择的合法性判断,如果输入不是 1 或 2,给予用户相应提示。
(3)在输入部分使用异常处理机制来捕获用户输入的异常情况,例如使用 try-catch
结构来处理 InputMismatchException
。
(4)在输出部分添加换行符,以提高用户体验。
题目:7-4 测验4-抽象类与接口
主要困难:(1)输入时没有对用户的非法输入进行检查,如输入非数字字符会导致程序出错。
(2)在创建图形对象时没有对用户输入的数据进行合法性检查,如半径小于等于 0、矩形的两个点重合等情况。
(3)在比较面积大小时,直接使用了 Comparable
接口和 compareTo
方法,虽然实现了排序,但没有灵活地处理排序方式。
改进建议:
(1)在用户输入部分,添加异常处理机制来捕获用户输入的异常情况,例如使用try-catch
结构来处理 InputMismatchException
。
(2)在创建图形对象时,对用户输入的数据进行合法性检查,如半径小于等于 0、矩形的两个点重合等情况,如果输入不合法,应给予用户相应提示并要求重新输入。
(3)在比较面积大小时,可以使用 Comparator
接口来创建自定义的比较器,以实现更灵活的排序方式,如按照面积升序或降序排列。
五、总结
我在本次大作业中学习到了许多东西:
首先,类结构设计是非常重要的,它可以帮助我们合理地组织和管理代码。在设计圆类的过程中,我学会了如何根据对象的属性和行为来设计类的成员变量和方法,以及如何使用访问修饰符来控制类的可访问性。这使得代码更加清晰和易于维护。
其次,我对继承与多态有了更深入的认识。在设计圆类的过程中,我意识到可以通过继承来实现代码的重用,并且可以通过多态来实现运行时的动态绑定,从而增强代码的灵活性和可扩展性。这使得我能够更好地应对需求变化和代码复用的情况。
此外,我也学习到了抽象类与接口的概念和用法。在设计圆类的过程中,我意识到抽象类可以作为其他类的基类,提供通用的属性和方法,而接口则可以定义类的行为规范,从而实现不同类之间的解耦合。这使得我能够更好地设计出灵活、可扩展的系统架构。
然而,我也意识到自己在以上方面还存在一些不足。比如在设计类结构时,我有时候会在一个类中包含过多的功能,导致类的职责不够单一;在使用继承与多态时,有时候会出现设计的层次过深或者过浅的情况,造成代码的可读性和可维护性下降;在使用抽象类与接口时,有时候会难以把握接口和抽象类的选择,导致系统架构的复杂度增加。
因此,我认为在今后的学习和工作中,我需要进一步加强对类结构设计、继承与多态、抽象类与接口等方面的理解,并且不断实践和总结经验,以提高自己的Java编程设计和开发能力。
标签:输出,int,Blog2,空格,菜品,21207328,吴义隆,table,输入 From: https://www.cnblogs.com/wuyilongblog1/p/17842472.html