前言
这是第二次博客作业,总结了近三次PTA大作业的完成情况,这三次的大作业难度逐渐增大,完全理不清逻辑,真的越想越混乱,代码写的也是很乱,没有一个整体的框架结构,读起来很困难,没有学到java程序编写的真谛,总之对于我,一个逻辑很差很差的人来说,越来越复杂的题目,写起来真的痛苦,到后面的题目,基本上就是毫无章法的加功能,为了题目而写题目,学起来好累TAT。
关于类和整个程序的设计:
学了很多新的结构设计的方法,和一些辅助清晰思路的方法,例如:画类图、画时序图,但是我还是不能很熟练的抽象出方法和各个类之间的联系,导致在编写复杂代码的时候会思维混乱不能有一个很好的代码结构。课堂上还学习了一些面向对象设计的原则,比如:里氏代换原则等,学习这些原则一定程度上帮助我在设计和编写代码的时候可以更加的规范和条理清晰。
三次题目集的知识点、题量、难度等情况:
(对于重难点题目将会给出完整题目)
第四次大作业:
7-4 菜单计价程序-2:这是点菜系列的第二次迭代,相比于前一次增加了可输入菜单等功能
设计点菜计价程序,根据输入的信息,计算并输出总价格。
输入内容按先后顺序包括两部分:菜单、订单,最后以"end"结束。
菜单由一条或多条菜品记录组成,每条记录一行
每条菜品记录包含:菜名、基础价格 两个信息。
订单分:点菜记录和删除信息。每一类信息都可包含一条或多条记录,每条记录一行。
点菜记录包含:序号、菜名、份额、份数。
份额可选项包括:1、2、3,分别代表小、中、大份。
删除记录格式:序号 delete
标识删除对应序号的那条点菜记录。
不同份额菜价的计算方法:
小份菜的价格=菜品的基础价格。
中份菜的价格=菜品的基础价格1.5。
小份菜的价格=菜品的基础价格2。
如果计算出现小数,按四舍五入的规则进行处理。
参考以下类的模板进行设计:
菜品类:对应菜谱上一道菜的信息。- Dish {
- String name;//菜品名称
- int unit_price; //单价
- int getPrice(int portion)//计算菜品价格的方法,输入参数是点菜的份额(输入数据只能是1/2/3,代表小/中/大份) }<br>
复制代码 菜谱类:对应菜谱,包含饭店提供的所有菜的信息。
- Menu {
- Dish[] dishs ;//菜品数组,保存所有菜品信息
- Dish searthDish(String dishName)//根据菜名在菜谱中查找菜品信息,返回Dish对象。
- Dish addDish(String dishName,int unit_price)//添加一道菜品信息
- }<br>
复制代码 点菜记录类:保存订单上的一道菜品记录
- Record {
- int orderNum;//序号\
- Dish d;//菜品\
- int portion;//份额(1/2/3代表小/中/大份)\
- int getPrice()//计价,计算本条记录的价格\
- }<br>
复制代码 订单类:保存用户点的所有菜的信息。- Order {
- Record[] records;//保存订单上每一道的记录
- int getTotalPrice()//计算订单的总价
- Record addARecord(int orderNum,String dishName,int portion,int num)//添加一条菜品信息到订单中。
- delARecordByOrderNum(int orderNum)//根据序号删除一条记录
- findRecordByNum(int orderNum)//根据序号查找一条记录
- }<br>
复制代码 输入格式:
菜品记录格式:
菜名+英文空格+基础价格
如果有多条相同的菜名的记录,菜品的基础价格以最后一条记录为准。
点菜记录格式:
序号+英文空格+菜名+英文空格+份额+英文空格+份数
注:份额可输入(1/2/3), 1代表小份,2代表中份,3代表大份。
删除记录格式:序号 +英文空格+delete
最后一条记录以“end”结束。
输出格式:
按顺序输出每条订单记录的处理信息,
每条点菜记录输出:序号+英文空格+菜名+英文空格+价格。其中的价格等于对应记录的菜品*份数,序号是之前输入的订单记录的序号。
如果订单中包含不能识别的菜名,则输出“** does not exist”,**是不能识别的菜名
如果删除记录的序号不存在,则输出“delete error”
最后输出订单上所有菜品的总价(整数数值),
本次题目不考虑其他错误情况,如:菜单订单顺序颠倒、不符合格式的输入、序号重复等。
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+英文空格+桌号+“:”+英文空格+当前桌的总价
本次题目不考虑其他错误情况,如:桌号、菜单订单顺序颠倒、不符合格式的输入、序号重复等,在本系列的后续作业中会做要求。
第五次大作业:
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+英文空格+桌号+“:”+英文空格+当前桌的原始总价+英文空格+当前桌的计算折扣后总价
第六次大作业:
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+英文空格+桌号+“:”+英文空格+当前桌的计算折扣后总价+英文空格+辣度平均值+英文空格+酸度平均值+英文空格+甜度平均值+英文空格
最后按拼音顺序输出每位客户(不考虑客户同名或拼音相同的情况)的支付金额,格式: 用户姓名+英文空格+手机号+英文空格+支付总金额,按输入顺序排列。
期中考试
期中考试考了类的基本设计、继承、多态、接口等,关于圆和四边形的类设计,我写题目的时候保存了一个编程题的题目,也是最后一个编程题,关于接口的,我编写的还是很不规范。- 在测验3的题目基础上,重构类设计,实现列表内图形的排序功能(按照图形的面积进行排序)。<br>提示:题目中Shape类要实现Comparable接口。<br><br>其中,Main类源码如下(可直接拷贝使用):<br><br>public class Main {<br> public static void main(String\[\] args) {<br> // TODO Auto-generated method stub<br> Scanner input = new Scanner(System.in);<br> ArrayList<已测试.Shape> list = new ArrayList<>();<br><br> int choice = input.nextInt();<br><br> while(choice != 0) {<br> switch(choice) {<br> case 1://已测试.Circle<br> double radiums = input.nextDouble();<br> 已测试.Shape circle = new 已测试.Circle(radiums);<br> list.add(circle);<br> break;<br> case 2://已测试.Rectangle<br> double x1 = input.nextDouble();<br> double y1 = input.nextDouble();<br> double x2 = input.nextDouble();<br> double y2 = input.nextDouble();<br> 已测试.Point leftTopPoint = new 已测试.Point(x1,y1);<br> 已测试.Point lowerRightPoint = new 已测试.Point(x2,y2);<br> 已测试.Rectangle rectangle = new 已测试.Rectangle(leftTopPoint,lowerRightPoint);<br> list.add(rectangle);<br> break;<br> }<br> choice = input.nextInt();<br> }<br><br> list.sort(Comparator.naturalOrder());//正向排序<br><br> for(int i = 0; i < list.size(); i++) {<br> System.out.print(String.format("%.2f", list.get(i).getArea()) + " ");<br> }<br> }<br>}<br><br>输入格式:<br>输入图形类型(1:圆形;2:矩形;0:结束输入)<br><br>输入图形所需参数<br><br>输出格式:<br>按升序排序输出列表中各图形的面积(保留两位小数),各图形面积之间用空格分隔。<br><br>输入样例:<br>在这里给出一组输入。例如:<br><br>1<br>2.3<br>2<br>3.2<br>3<br>6<br>5<br>1<br>2.3<br>0<br>输出样例:<br>在这里给出相应的输出。例如:<br><br>5.60 16.62 16.62
复制代码 设计与分析与主要问题和其解决方法
(这里只会提供重点题目的代码)
第四次大作业:
菜单2
- 1 import java.math.BigDecimal;
- 2 import java.util.HashMap;
- 3 import java.util.Map;
- 4
- 5 class Dish {
- 6 String name;
- 7 int unit_price;
- 8
- 9 public Dish(String name, int unit_price) {
- 10 this.name = name;
- 11 this.unit_price = unit_price;
- 12 }
- 13
- 14 public long getPrice(int portion) {
- 15 double price = unit_price;
- 16 if (portion == 2) {
- 17 price *= 1.5;
- 18 } else if (portion == 3) {
- 19 price *= 2;
- 20 }
- 21 return Math.round(price);
- 22 }
- 23 }
- 24
- 25 class Menu {
- 26 static Dish[] dishes;
- 27
- 28 public Dish addDish(String dishName, int unit_price) {
- 29 Dish dish = new Dish(dishName, unit_price);
- 30 for (int i = 0; i < dishes.length; i++) {
- 31 if (dishes[i] != null && dishes[i].name.equals(dishName)) {
- 32 dishes[i] = dish;
- 33 return dish;
- 34 }
- 35 }
- 36 // 如果未找到相同菜名的菜品记录,则添加到新的位置
- 37 for (int i = 0; i < dishes.length; i++) {
- 38 if (dishes[i] == null) {
- 39 dishes[i] = dish;
- 40 break;
- 41 }
- 42 }
- 43 return dish;
- 44 }
- 45
- 46 public static Dish searchDish(String dishName) {
- 47 for (Dish dish : dishes) {
- 48 if (dish != null && dish.name.equals(dishName)) {
- 49 return dish;
- 50 }
- 51 }
- 52 return null;
- 53 }
- 54 }
- 55
- 56 class Record {
- 57 int orderNum;
- 58 Dish dish;
- 59 int portion;
- 60
- 61 public long getPrice() {
- 62 if(dish != null)
- 63 return dish.getPrice(portion);
- 64 else return 0;
- 65 }
- 66 }
- 67
- 68 class Order {
- 69 Record[] records;
- 70
- 71 public Record addARecord(int orderNum, String dishName, int portion, int num) {
- 72 Record record = new Record();
- 73 record.orderNum = orderNum;
- 74 record.dish = Menu.searchDish(dishName);
- 75 record.portion = portion;
- 76
- 77 int index = 0;
- 78 while (index < records.length && records[index] != null) {
- 79 index++;
- 80 }
- 81
- 82 if (index < records.length) {
- 83 records[index] = record;
- 84 } else {
- 85 System.out.println("记录已满,无法添加新的点菜记录。");
- 86 }
- 87 return record;
- 88 }
- 89
- 90 public void delARecordByOrderNum(int orderNum) {
- 91 for (int i = 0; i < records.length; i++) {
- 92 if (records[i] != null && records[i].orderNum == orderNum) {
- 93 records[i] = null;
- 94 break;
- 95 }
- 96 }
- 97 }
- 98
- 99 public Record findRecordByNum(int orderNum) {
- 100 for (Record record : records) {
- 101 if (record != null && record.orderNum == orderNum) {
- 102 return record;
- 103 }
- 104 }
- 105 return null;
- 106 }
- 107
- 108 public long getTotalPrice() {
- 109 long totalPrice = 0;
- 110 for (Record record : records) {
- 111 if (record != null) {
- 112 totalPrice =totalPrice + record.getPrice();
- 113 }
- 114 }
- 115 return totalPrice;
- 116 }
- 117 }
- 118
- 119
- 120 public class PTA4_4_order2 {
- 121 private static Map<Integer,BigDecimal> number= new HashMap<Integer,BigDecimal>();
- 122 public static void main(String[] args) {
- 123 BigDecimal bigDecimal = new BigDecimal(0L);
- 124 Menu menu = new Menu();
- 125 menu.dishes = new Dish[100];
- 126
- 127 Order order = new Order();
- 128 order.records = new Record[100];
- 129
- 130 java.util.Scanner scanner = new java.util.Scanner(System.in);
- 131 String input = scanner.nextLine();
- 132 while (true) {
- 133 String[] splitInput = input.split(" ");
- 134 // 处理菜单记录
- 135 if (splitInput.length == 2) {
- 136 String dishName = splitInput[0];
- 137 String PO = splitInput[1];
- 138 if (!PO.equals("delete")){
- 139 int unitPrice = Integer.parseInt(splitInput[1]);
- 140 menu.addDish(dishName, unitPrice);
- 141 }else {
- 142 int orderNum = Integer.parseInt(splitInput[0]);
- 143 Record record = order.findRecordByNum(orderNum);
- 144 BigDecimal o = number.get(orderNum);
- 145 if (o != null){
- 146 bigDecimal = bigDecimal.subtract(o);
- 147 }
- 148 if (record == null) {
- 149 System.out.println("delete error;");
- 150 } else {
- 151 order.delARecordByOrderNum(orderNum);
- 152 }
- 153 }
- 154 }
- 155 // 处理订单记录
- 156 else if (splitInput.length == 4) {
- 157 int orderNum = Integer.parseInt(splitInput[0]);
- 158 String dishName = splitInput[1];
- 159 int portion = Integer.parseInt(splitInput[2]);
- 160 int num = Integer.parseInt(splitInput[3]);
- 161 Record record = order.addARecord(orderNum, dishName, portion, num);
- 162 if (record.dish == null) {
- 163 System.out.println(dishName + " does not exist");
- 164 } else {
- 165 long totalPrice = record.getPrice() * num;
- 166 BigDecimal bigDecimal1 = BigDecimal.valueOf(totalPrice);
- 167 number.put(orderNum,bigDecimal1);
- 168 bigDecimal = bigDecimal.add(bigDecimal1);
- 169 System.out.println(orderNum + " " + dishName + " " + totalPrice);
- 170 }
- 171 }
- 172 // 处理删除记录
- 173 else if (splitInput[1].equals("delete")) {
- 174 int orderNum = Integer.parseInt(splitInput[0]);
- 175
- 176 Record record = order.findRecordByNum(orderNum);
- 177 if (record == null) {
- 178 System.out.println("delete error");
- 179 } else {
- 180 order.delARecordByOrderNum(orderNum);
- 181 }
- 182 }
- 183 input = scanner.nextLine();
- 184 if (input.equals("end")){
- 185 break;
- 186 }
- 187 }
- 188 scanner.close();
- 189 System.out.print(bigDecimal.setScale(1, BigDecimal.ROUND_HALF_UP).intValue());
- 190 }
- 191 }
复制代码 View Code类图如下:
题目的分析如下:
这个题目需要设计一个点菜计价程序,可以根据输入的菜单和订单信息,计算并输出总价格。首先,你需要设计三个类:菜品类(Dish)、菜谱类(Menu)、点菜记录类(Record)和订单类(Order)。
菜品类(Dish)包含菜品名称和基础价格,以及计算菜品价格的方法。菜谱类(Menu)包含了一个菜品数组,可以根据菜名在菜谱中查找菜品信息,或者添加一道菜品信息。点菜记录类(Record)保存订单上的一道菜品记录,包含序号、菜品、份额和计价方法。订单类(Order)保存用户点的所有菜的信息,包括订单上的每一道记录,计算订单的总价,添加一条菜品信息到订单中以及根据序号删除一条记录。
输入格式包括菜单和订单,以"end"结束。菜单的记录格式为菜名+英文空格+基础价格,而点菜记录格式为序号+英文空格+菜名+英文空格+份额+英文空格+份数。删除记录格式为序号+英文空格+delete。输出格式包括每条订单记录的处理信息,然后输出订单上所有菜品的总价(整数数值)。
需要注意的是,如果订单中包含不能识别的菜名,则输出“** does not exist”,如果删除记录的序号不存在,则输出“delete error”。
在程序中,需要对输入进行解析,创建菜单和订单对象,并根据订单进行计价。最后,输出每条订单记录的处理信息和订单的总价。
菜单3
[code] 1 import java.math.BigDecimal; 2 import java.time.LocalDate; 3 import java.time.format.DateTimeFormatter; 4 import java.time.format.DateTimeParseException; 5 import java.util.Date; 6 import java.util.HashMap; 7 import java.util.Map; 8 import java.util.Timer; 9 import java.time.DayOfWeek; 10 import java.time.LocalTime; 11 12 class Dish { 13 String name; 14 int unit_price; 15 16 public Dish(String name, int unit_price) { 17 this.name = name; 18 this.unit_price = unit_price; 19 } 20 21 public long getPrice(int portion) { 22 double price = unit_price; 23 if (portion == 2) { 24 price *= 1.5; 25 } else if (portion == 3) { 26 price *= 2; 27 } 28 return Math.round(price); 29 } 30 } 31 32 class Menu { 33 static Dish[] dishes; 34 35 public Dish addDish(String dishName, int unit_price) { 36 Dish dish = new Dish(dishName, unit_price); 37 for (int i = 0; i < dishes.length; i++) { 38 if (dishes != null && dishes.name.equals(dishName)) { 39 dishes = dish; 40 return dish; 41 } 42 } 43 // 如果未找到相同菜名的菜品记录,则添加到新的位置 44 for (int i = 0; i < dishes.length; i++) { 45 if (dishes == null) { 46 dishes = dish; 47 break; 48 } 49 } 50 return dish; 51 } 52 53 public static Dish searchDish(String dishName) { 54 for (Dish dish : dishes) { 55 if (dish != null && dish.name.equals(dishName)) { 56 return dish; 57 } 58 } 59 return null; 60 } 61 } 62 63 class Record { 64 int orderNum; 65 Dish dish; 66 int portion; 67 68 public long getPrice() { 69 if(dish != null) 70 return dish.getPrice(portion); 71 else return 0; 72 } 73 } 74 75 class Order { 76 Record[] records; 77 78 public Record addARecord(int orderNum, String dishName, int portion, int num) { 79 Record record = new Record(); 80 record.orderNum = orderNum; 81 record.dish = Menu.searchDish(dishName); 82 record.portion = portion; 83 84 int index = 0; 85 while (index < records.length && records[index] != null) { 86 index++; 87 } 88 89 if (index < records.length) { 90 records[index] = record; 91 } else { 92 System.out.println("记录已满,无法添加新的点菜记录。"); 93 } 94 return record; 95 } 96 97 public void delARecordByOrderNum(int orderNum) { 98 for (int i = 0; i < records.length; i++) { 99 if (records != null && records.orderNum == orderNum) {100 records = null;101 break;102 }103 }104 }105 106 public Record findRecordByNum(int orderNum) {107 for (Record record : records) {108 if (record != null && record.orderNum == orderNum) {109 return record;110 }111 }112 return null;113 }114 115 public long getTotalPrice() {116 long totalPrice = 0;117 for (Record record : records) {118 if (record != null) {119 totalPrice =totalPrice + record.getPrice();120 }121 }122 return totalPrice;123 }124 }125 126 class Table{127 static int tableNum;128 static Order order;129 Date day;130 Timer time;131 // 验证日期是否合法132 public static boolean isDateValid(String date1) {133 try {134 // 解析日期字符串135 LocalDate.parse(date1);136 return true; // 如果解析成功,则日期合法137 } catch (DateTimeParseException e) {138 return false; // 如果解析失败,则日期非法139 }140 }141 142 public static boolean isWeekend(String dateString, String format) {143 try {144 DateTimeFormatter formatter = DateTimeFormatter.ofPattern(format);145 LocalDate date = LocalDate.parse(dateString, formatter);146 DayOfWeek dayOfWeek = date.getDayOfWeek();147 return dayOfWeek == DayOfWeek.SATURDAY || dayOfWeek == DayOfWeek.SUNDAY;148 } catch (Exception e) {149 return false; // 日期格式不正确150 }151 }152 153 //判断是否在周末营业时间154 public static boolean isWithinTimeRange(String timeString, String format) {155 try {156 DateTimeFormatter formatter = DateTimeFormatter.ofPattern(format);157 LocalTime time = LocalTime.parse(timeString, formatter);158 LocalTime lowerBound = LocalTime.parse("09/30/00", formatter);159 LocalTime upperBound = LocalTime.parse("21/30/00", formatter);160 return lowerBound.compareTo(time) |