知识点:字符串处理、面向对象、对于数据保存的方法选择以及对不同类之间关系的处理、异常处理、数组、字符串操作、四舍五入、计算折扣
7-1测验1-圆类设计
创建一个圆形类(Circle),私有属性为圆的半径,从控制台输入圆的半径,输出圆的面积
输入圆的半径,取值范围为(0,+∞),输入数据非法,则程序输出WrongFormat,注意:只考虑从控制台输入数值的情况
输出圆的面积(保留两位小数,可以使用String.format(“%.2f”,输出数值)控制精度)
在这里给出一组输入。例如:
2.35输出样例:在这里给出相应的输出。例如:
17.35importjava.util.Scanner;publicclassMain{publicstaticvoidmain(String[]args){Scannerin=newScanner(System.in);Circleround=newCircle();doubler=in.nextDouble();round.Area(r);}}classCircle{privatedoubler;publicvoidArea(doubler){if(r>0){doublepi=Math.PI;Stringstr=null;str=String.format("%.2f",pi*r*r);System.out.println(""+str);}else{System.out.println("WrongFormat");}}}2.采坑心得:
关键在于PI的引入,保证输出值可以过测试点;另外学到了控制精度的方法。
7-2测验2-类结构设计
设计一个矩形类,其属性由矩形左上角坐标点(x1,y1)及右下角坐标点(x2,y2)组成,其中,坐标点属性包括该坐标点的X轴及Y轴的坐标值(实型数),求得该矩形的面积。类设计如下图:
分别输入两个坐标点的坐标值x1,y1,x2,y2。
输出该矩形的面积值(保留两位小数)。
65.8-78.9输出样例:在这里给出相应的输出。例如:
set方法用于无参构造,get方法用于封装的属性的外部访问。先构造点再构造矩形。
但是最后输出结果无法通过输出正确的测试点,可能是输出精度控制不正确的原因,
7-3测验3-继承与多态
将测验1与测验2的类设计进行合并设计,抽象出Shape父类(抽象类),Circle及Rectangle作为子类,类图如下所示:
试编程完成如上类图设计,主方法源码如下(可直接拷贝使用):
publicstaticvoidmain(String[]args){//TODOAuto-generatedmethodstubScannerinput=newScanner(System.in);intchoice=input.nextInt();switch(choice){case1://Circledoubleradiums=input.nextDouble();Shapecircle=newCircle(radiums);printArea(circle);break;case2://Rectangledoublex1=input.nextDouble();doubley1=input.nextDouble();doublex2=input.nextDouble();doubley2=input.nextDouble();PointleftTopPoint=newPoint(x1,y1);PointlowerRightPoint=newPoint(x2,y2);Rectanglerectangle=newRectangle(leftTopPoint,lowerRightPoint);printArea(rectangle);break;}}其中,printArea(Shapeshape)方法为定义在Main类中的静态方法,体现程序设计的多态性。
输入类型选择(1或2,不考虑无效输入)对应图形的参数(圆或矩形)
图形的面积(保留两位小数)
15.6输出样例1:在这里给出相应的输出。例如:
98.52输入样例2:25.6-32.59.4-5.6输出样例2:在这里给出相应的输出。例如:
继承有一个特性重写,就是说子类方法中可以有和父类同名的方法但是执行内容不同,父类方法中子类有重写则父类中该方法可以不写操作(有返回值则返回0,无则不写),再者由main方法里的步骤可知子类只需调用有参构造因此果断删除无参构造,改写前两题的代码,然后父类中必有无参构造器,子类构造器默认调用父类中的无参构造(这样编译才会通过)。prinArea()方法为Main类中的static方法则按要求加上即可(?多态性?)。
7-4测验4-抽象类与接口
在测验3的题目基础上,重构类设计,实现列表内图形的排序功能(按照图形的面积进行排序)。提示:题目中Shape类要实现Comparable接口。
其中,Main类源码如下(可直接拷贝使用):
publicclassMain{publicstaticvoidmain(String\[\]args){//TODOAuto-generatedmethodstubScannerinput=newScanner(System.in);ArrayList 输入图形所需参数 按升序排序输出列表中各图形的面积(保留两位小数),各图形面积之间用空格分隔。 12.323.236512.30输出样例:在这里给出相应的输出。例如: 5.6016.6216.62 7-1菜单计价程序-4 本体大部分内容与菜单计价程序-3相同,增加的部分用加粗文字进行了标注。 设计点菜计价程序,根据输入的信息,计算并输出总价格。 输入内容按先后顺序包括两部分:菜单、订单,最后以"end"结束。 菜单由一条或多条菜品记录组成,每条记录一行 每条菜品记录包含:菜名、基础价格两个信息。 订单分:桌号标识、点菜记录和删除信息、代点菜信息。每一类信息都可包含一条或多条记录,每条记录一行或多行。 桌号以下的所有记录都是本桌的记录,直至下一个桌号标识。 点菜记录包含:序号、菜名、份额、份数。份额可选项包括:1、2、3,分别代表小、中、大份。 不同份额菜价的计算方法:小份菜的价格=菜品的基础价格。中份菜的价格=菜品的基础价格1.5。小份菜的价格=菜品的基础价格2。如果计算出现小数,按四舍五入的规则进行处理。 删除记录格式:序号delete 标识删除对应序号的那条点菜记录。 如果序号不对,输出"deleteerror" 代点菜信息包含:桌号序号菜品名称份额分数 代点菜是当前桌为另外一桌点菜,信息中的桌号是另一桌的桌号,带点菜的价格计算在当前这一桌。 程序最后按输入的桌号从小到大的顺序依次输出每一桌的总价(注意:由于有代点菜的功能,总价不一定等于当前桌上的菜的价格之和)。 每桌的总价等于那一桌所有菜的价格之和乘以折扣。如存在小数,按四舍五入规则计算,保留整数。 参考以下类的模板进行设计(本内容与计价程序之前相同,其他类根据需要自行定义): 菜品类:对应菜谱上一道菜的信息。 Dish{ Stringname;//菜品名称 intunit_price;//单价 intgetPrice(intportion)//计算菜品价格的方法,输入参数是点菜的份额(输入数据只能是1/2/3,代表小/中/大份)} 菜谱类:对应菜谱,包含饭店提供的所有菜的信息。 Menu{ Dish[]dishs;//菜品数组,保存所有菜品信息 DishsearthDish(StringdishName)//根据菜名在菜谱中查找菜品信息,返回Dish对象。 DishaddDish(StringdishName,intunit_price)//添加一道菜品信息 } 点菜记录类:保存订单上的一道菜品记录 Record{ intorderNum;//序号 Dishd;//菜品\\ intportion;//份额(1/2/3代表小/中/大份) intgetPrice()//计价,计算本条记录的价格 订单类:保存用户点的所有菜的信息。 Order{ Record[]records;//保存订单上每一道的记录 intgetTotalPrice()//计算订单的总价 RecordaddARecord(intorderNum,StringdishName,intportion,intnum)//添加一条菜品信息到订单中。 delARecordByOrderNum(intorderNum)//根据序号删除一条记录 findRecordByNum(intorderNum)//根据序号查找一条记录 本次课题比菜单计价系列-3增加的异常情况: 1、菜谱信息与订单信息混合,应忽略夹在订单信息中的菜谱信息。输出:"invaliddish" 3、同一桌菜名、份额相同的点菜记录要合并成一条进行计算,否则可能会出现四舍五入的误差。 4、重复删除,重复的删除记录输出"deduplication:"+序号。 6、菜谱信息中出现重复的菜品名,以最后一条记录为准。 9、份额超出范围(1、2、3)输出:序号+"portionoutofrange"+份额,份额不能超过1位,否则为非法格式,参照第13条输出。 10、份数超出范围,每桌不超过15份,超出范围输出:序号+"numoutofrange"+份数。份数必须为数值,最高位不能为0,否则按非法格式参照第16条输出。 11、桌号超出范围[1,55]。输出:桌号+"tablenumoutofrange",桌号必须为1位或多位数值,最高位不能为0,否则按非法格式参照第16条输出。 12、菜谱信息中菜价超出范围(区间(0,300)),输出:菜品名+"priceoutofrange"+价格,菜价必须为数值,最高位不能为0,否则按非法格式参照第16条输出。 14、一条点菜记录中若格式正确,但数据出现问题,如:菜名不存在、份额超出范围、份数超出范围,按记录中从左到右的次序优先级由高到低,输出时只提示优先级最高的那个错误。 15、每桌的点菜记录的序号必须按从小到大的顺序排列(可以不连续,也可以不从1开始),未按序排列序号的输出:"recordserialnumbersequenceerror"。当前记录忽略。(代点菜信息的序号除外) 16、所有记录其它非法格式输入,统一输出"wrongformat" 17、如果记录以“table”开头,对应记录的格式或者数据不符合桌号的要求,那一桌下面定义的所有信息无论正确或错误均忽略,不做处理。如果记录不是以“table”开头,比如“table552023/3/212/00/00”,该条记录认为是错误记录,后面所有的信息并入上一桌一起计算。 本次作业比菜单计价系列-3增加的功能: 菜单输入时增加特色菜,特色菜的输入格式:菜品名+英文空格+基础价格+"T" 例如:麻婆豆腐9T 菜价的计算方法: 周一至周五7折,周末全价。 注意:不同的四舍五入顺序可能会造成误差,请按以下步骤累计一桌菜的菜价: 计算每条记录的菜价:将每份菜的单价按份额进行四舍五入运算后,乘以份数计算多份的价格,然后乘以折扣,再进行四舍五入,得到本条记录的最终支付价格。 最后将所有记录的菜价累加得到整桌菜的价格。 菜品记录格式: 菜名+英文空格+基础价格 如果有多条相同的菜名的记录,菜品的基础价格以最后一条记录为准。 点菜记录格式:序号+英文空格+菜名+英文空格+份额+英文空格+份数注:份额可输入(1/2/3),1代表小份,2代表中份,3代表大份。 删除记录格式:序号+英文空格+delete 代点菜信息包含:桌号+英文空格+序号+英文空格+菜品名称+英文空格+份额+英文空格+分数 最后一条记录以“end”结束。 按输入顺序输出每一桌的订单记录处理信息,包括: 1、桌号,格式:table+英文空格+桌号+”:”+英文空格 2、按顺序输出当前这一桌每条订单记录的处理信息, 每条点菜记录输出:序号+英文空格+菜名+英文空格+价格。其中的价格等于对应记录的菜品*份数,序号是之前输入的订单记录的序号。如果订单中包含不能识别的菜名,则输出“**doesnotexist”,**是不能识别的菜名 如果删除记录的序号不存在,则输出“deleteerror” 最后按输入顺序一次输出每一桌所有菜品的总价(整数数值)格式:table+英文空格+桌号+“:”+英文空格+当前桌的原始总价+英文空格+当前桌的计算折扣后总价 麻婆豆腐12油淋生菜9Ttable312023/2/114/20/001麻婆豆腐1162油淋生菜122delete2deleteend输出样例:在这里给出相应的输出。例如: table31:1numoutofrange162油淋生菜18deduplication2table31:00输入样例1:份数超出范围+份额超出范围。例如: 麻婆豆腐12油淋生菜9Ttable312023/2/114/20/001麻婆豆腐1162油淋生菜42end输出样例1:份数超出范围+份额超出范围。例如: table31:1numoutofrange162portionoutofrange4table31:00输入样例2:桌号信息错误。例如: 麻婆豆腐12油淋生菜9Ttablea2023/3/1512/00/001麻婆豆腐112油淋生菜21end输出样例2:在这里给出相应的输出。例如: wrongformat输入样例3:混合错误:桌号信息格式错误+混合的菜谱信息(菜谱信息忽略)。例如: 麻婆豆腐12油淋生菜9Ttable552023/3/3112/000/00麻辣香锅151麻婆豆腐112油淋生菜21end输出样例3:在这里给出相应的输出。例如: wrongformat输入样例4:错误的菜谱记录。例如: 麻婆豆腐12.0油淋生菜9Ttable552023/3/3112/00/00麻辣香锅151麻婆豆腐112油淋生菜21end输出样例4:在这里给出相应的输出。例如: wrongformattable55:invaliddish麻婆豆腐doesnotexist2油淋生菜14table55:1410输入样例5:桌号格式错误(以“table”开头)+订单格式错误(忽略)。例如: 麻婆豆腐12油淋生菜9Ttablea2023/3/1512/00/001麻婆豆腐112油淋生菜21end输出样例5:在这里给出相应的输出。例如: wrongformat输入样例6:桌号格式错误,不以“table”开头。例如: 麻婆豆腐12油淋生菜9Ttable12023/3/1512/00/001麻婆豆腐112油淋生菜21table22023/3/1512/00/001麻婆豆腐112油淋生菜21end输出样例6:在这里给出相应的输出。例如: 分析: 本次作业设计的难点之一是处理各种异常情况。在设计程序时,需要考虑到各种不合法的输入格式和数据,对其进行逐个判断和处理,输出对应的异常提示信息。另外,菜价计算的过程也需要注意精度问题,例如四舍五入的实现方式、累加时的精度控制等。 在本次作业的实现过程中,我的思路是先定义好各种类的属性和方法,并且按照输入信息的顺序逐个处理,然后统一输出结果。在处理输入信息时,要细心、严谨,对于每一个不合法的输入都要及时输出提示信息,同时记录错误信息,以便修改和调试程序。 在编写代码时,我还使用了一些代码规范和优化的技巧,例如使用try-catch捕获异常,采用命名规范和代码注释规范等。通过这次作业的设计和实现,我加深了对Java程序设计的理解和应用,提高了自己的编程能力和实践经验。 踩坑心得: 改进建议: 1、对于部分输入格式不合法的情况,建议输出具体的出错信息,便于排查问题。 2、对于代点菜的功能,建议增加输出代点菜的桌号和被代点菜的桌号,以便进行核对。 3、计价时建议将每一条记录的小计输出,方便核对。 4、对于同一桌重复的点菜记录,建议在添加新记录时进行合并,而不是在计价时进行合并。 5、建议在输入订单记录时对序号进行校验,确保序号的连续性和正确性。 6、计算菜价时建议使用BigDecimal类进行精确计算,避免浮点数计算误差。 总结: 通过这道作业,我了解了Java中异常处理的方法,并运用到了实际项目中。 在此过程中,我发现了自己对于异常处理的不熟练程度,需要更多的实践进行加强。 同时,作为一道相对复杂的作业,也让我更深刻地认识到了良好的代码设计和规范的重要性,代码的可读性和可维护性对于一个项目的成功实现有着至关重要的作用。 对于教师、课程、作业、实验、课上及课下组织方式等方面,我认为除了能够掌握更多的知识之外,更为重要的是能够培养我们独立思考和解决问题的能力,这才是我们未来发展所必须的能力。同时,我也希望有更多的实践机会来加强自己的能力,并更好地掌握所学知识。