首页 > 其他分享 >BLOG-2

BLOG-2

时间:2023-11-19 23:44:06浏览次数:32  
标签:空格 记录 int line2 BLOG 菜品 table

一、前言

  这次是对PTA题目集4、5、6以及期中考试的总结性Blog。第4、5、6次PTA的主要内容就是对于菜单系列的题目的迭代。一次比一次复杂了。第4次题目集主要是菜单2和菜单3。当时写起来觉得很难,但是后面写了,感觉跟后面的相比之下还是容易一点。然后第5、6次题目集都是一次只有一题了。虽然题量少了很多,但是难度大大增加了,测试点也多了很多,特别是看不见样例的测试点,过不去也找不到为什么,真的很难受。通过率也一下就减少了很多。不及格的人也多了。而期中考试主要是二十多个选择题,考一些理论知识。还有几个编程题,主要考继承和多态的用法。

二、设计与分析

 第4次题目集:菜单计价程序-2

  1、题目

设计点菜计价程序,根据输入的信息,计算并输出总价格。

输入内容按先后顺序包括两部分:菜单、订单,最后以"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

  2、源码

import java.util.Scanner;
import java.util.Vector;

public class Main {
	public static void main(String[] args) {
		Scanner input = new Scanner(System.in);
		Menu menu = new Menu();
		Order order = new Order();
		while (input.hasNextLine()) {  //当有输入时
			String line1 = input.nextLine();
			if (line1.equals("end"))  //遇见end结束
				break;
			String[] line2 = line1.split(" ");
			if (line2.length == 2) {
				String word1 = line2[0];
				String word2 = line2[1];
				if (word1.matches("^\\d*$")) {  //删除记录
					int deleteNum = Integer.parseInt(word1);
					order.delARecordByOrderNum(deleteNum);
				}
				else {  //菜单添加菜品信息
					menu.addDish(word1, Integer.parseInt(word2));
				}
			}
			else if(line2.length == 4) {  //添加点菜记录
				int orderNum = Integer.parseInt(line2[0]);
				String dishName = line2[1];
				int portion = Integer.parseInt(line2[2]);
				int num = Integer.parseInt(line2[3]);
				order.addARecord(orderNum, dishName, portion, num, menu);
			}
		}
		System.out.print(order.getTotalPrice());
	}
}

// 菜品类:对应菜谱上一道菜的信息。
class Dish {
	String name;// 菜品名称
	int unit_price; // 单价
	Dish(String name,int unit_price) {
		this.name = name;
		this.unit_price = unit_price;
	}
	int getPrice(int portion) {// 计算菜品价格的方法,输入参数是点菜的份额(输入数据只能是1/2/3,代表小/中/大份)
		if(portion == 1)
			return unit_price;
		else if (portion == 2)
			return (int)Math.round(unit_price * 1.5);
		else if (portion == 3)
			return unit_price *2;
		else
			return 0;
	}
}

//菜谱类:对应菜谱,包含饭店提供的所有菜的信息。
class Menu {
	Vector<Dish> dishes = new Vector<>();// 菜品数组,保存所有菜品信息
	Dish searthDish(String dishName) { // 根据菜名在菜谱中查找菜品信息,返回Dish对象。
		for (Dish dish:dishes)
			if (dish.name.equals(dishName))
				return dish;
		return null;
	}
	void addDish(String dishName, int unit_price) { // 添加一道菜品信息
		Dish dish = new Dish(dishName,unit_price);
		for (int i=0;i<dishes.size();i++) {
			if (dishes.get(i).name.equals(dishName)) {  //如果有多条相同的菜名的记录,菜品的基础价格以最后一条记录为准。
				dishes.set(i, dish);
				return;
			}
		}
		dishes.add(dish);
	}
}

//点菜记录类:保存订单上的一道菜品记录
class Record {
	int orderNum;// 序号
	Dish d;// 菜品
	int portion;// 份额(1/2/3代表小/中/大份)
	int num;  //份数
	Record(int orderNum, Dish d, int portion, int num) {
		this.orderNum = orderNum;
		this.d = d;
		this.portion = portion;
		this.num = num;
	}
	int getPrice() {// 计价,计算本条记录的价格
		return d.getPrice(portion) * num;
	}
}

//订单类:保存用户点的所有菜的信息。
class Order {
	Vector<Record> records = new Vector<>();// 保存订单上每一道的记录
	int getTotalPrice() { // 计算订单的总价
		int sum = 0;
		for (Record record:records)
			sum += record.getPrice();
		return sum;
	}
	void addARecord(int orderNum, String dishName, int portion, int num,Menu menu) {// 添加一条菜品信息到订单中。
		Dish d = menu.searthDish(dishName);
		if (d != null) {  //菜品存在时
			Record record = new Record(orderNum,d,portion,num);
			records.add(record);
			System.out.println(record.orderNum + " "  +dishName + " " + record.getPrice());
		}
		else {		//菜品不存在时
			Dish dish = new Dish(dishName,0);
			Record record = new Record(orderNum,dish,0,0);
			records.add(record);
			System.out.println(dishName+ " does not exist");
		}
	}
	void delARecordByOrderNum(int orderNum) {//根据序号删除一条记录
		Record record = findRecordByNum(orderNum);
		if (record == null)
			System.out.println("delete error;");
		else
			records.remove(record);
	}
	Record findRecordByNum(int orderNum) {//根据序号查找一条记录
		for (Record record:records) {
			if (record.orderNum == orderNum)
				return record;
		}
		return null;	
	}
}

  3、类图

  4、分析

  这次相对来说不算难,可是我一开始卡了很久,看了大概三四天没看出来有什么问题,最后还是朋友帮看出来就是一个四舍五入的问题,因为偷懒没有用Math.round函数,导致许多结果有问题,现在想来当时菜单1一直过不去应该也就是这个四舍五入的问题。很气自己怎么就偷懒了。浪费很多时间。

 

第4次题目集:菜单计价程序-3

  1、题目

设计点菜计价程序,根据输入的信息,计算并输出总价格。

输入内容按先后顺序包括两部分:菜单、订单,最后以"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

  2、源码

import java.util.*;
import java.time.DayOfWeek;
import java.time.LocalDate;
import java.time.format.DateTimeFormatter;

public class Main {
	public static void main(String[] args) {
		Scanner input = new Scanner(System.in);
		Menu menu = new Menu();
		Table[] table = new Table[100];  //桌数
		int index = -1;  //table数组的序号
		DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy/M/d"); // 定义日期时间格式
		while (input.hasNextLine()) {  //当有输入时
			String line1 = input.nextLine();
			if (line1.equals("end"))  //遇见end结束
				break;
			String[] line2 = line1.split(" ");
			if (line2.length == 2) {
				String word1 = line2[0];
				String word2 = line2[1];
				if (word1.matches("^\\d*$")) {  //删除记录
					int deleteNum = Integer.parseInt(word1);
					table[index].order.delARecordByOrderNum(deleteNum);
				}
				else {  //菜单添加菜品信息
					menu.addDish(word1, Integer.parseInt(word2));
				}
			}
			else if(line2.length == 4) {  
				if (line2[0].equals("table")) { //新增一桌信息
					int tableNum = Integer.parseInt(line2[1]);
					System.out.println("table " + tableNum + ": ");
					LocalDate date = LocalDate.parse(line2[2], formatter); // 将输入日期的字符串形式转换为刚刚定义的日期时间格式
					String[] time = line2[3].split("/");
					int hour = Integer.parseInt(time[0]);
					int minute = Integer.parseInt(time[1]);
					table[++index] = new Table(tableNum,date,hour,minute);
				}
				else {	//自己桌点菜,添加点菜记录
					int orderNum = Integer.parseInt(line2[0]);
					String dishName = line2[1];
					int portion = Integer.parseInt(line2[2]);
					int num = Integer.parseInt(line2[3]);
					table[index].order.addASelfRecord(orderNum, dishName, portion, num, menu);
				}
			}
			else if (line2.length == 5) {  //替其他桌点菜,添加点菜记录
				int orderNum = Integer.parseInt(line2[1]);
				String dishName = line2[2];
				int portion = Integer.parseInt(line2[3]);
				int num = Integer.parseInt(line2[4]);
				table[index].order.addAOtherRecord(orderNum, dishName, portion, num, menu, table[index], Integer.parseInt(line2[0]));
			}
		}
		for (int i=0;i<=index;i++) {  //计算总价
			String discount = table[i].judgeTime();  //根据时间得到折扣
			if (discount.equals("no"))
				System.out.println("table " + table[i].tableNum + " out of opening hours");
			else 
				System.out.println("table " + table[i].tableNum + ": " + table[i].order.getTotalPrice(discount));
		}
	}
}

//菜品类:对应菜谱上一道菜的信息。
class Dish {
	String name;// 菜品名称
	int unit_price; // 单价
	Dish(String name,int unit_price) {
		this.name = name;
		this.unit_price = unit_price;
	}
	int getPrice(int portion) {// 计算菜品价格的方法,输入参数是点菜的份额(输入数据只能是1/2/3,代表小/中/大份)
		if(portion == 1)
			return unit_price;
		else if (portion == 2)
			return (int)Math.round(unit_price * 1.5);
		else if (portion == 3)
			return unit_price *2;
		else
			return 0;
	}
}

//菜谱类:对应菜谱,包含饭店提供的所有菜的信息。
class Menu {
	Vector<Dish> dishes = new Vector<>();// 菜品数组,保存所有菜品信息
	Dish searthDish(String dishName) { // 根据菜名在菜谱中查找菜品信息,返回Dish对象。
		for (Dish dish:dishes)
			if (dish.name.equals(dishName))
				return dish;
		return null;
	}
	void addDish(String dishName, int unit_price) { // 添加一道菜品信息
		Dish dish = new Dish(dishName,unit_price);
		for (int i=0;i<dishes.size();i++) {
			if (dishes.get(i).name.equals(dishName)) {  //如果有多条相同的菜名的记录,菜品的基础价格以最后一条记录为准。
				dishes.set(i, dish);
				return;
			}
		}
		dishes.add(dish);
	}
}

//点菜记录类:保存订单上的一道菜品记录
class Record {
	int orderNum;// 序号
	Dish d;// 菜品
	int portion;// 份额(1/2/3代表小/中/大份)
	int num;  //份数
	Record(int orderNum, Dish d, int portion, int num) {
		this.orderNum = orderNum;
		this.d = d;
		this.portion = portion;
		this.num = num;
	}
	int getPrice() {// 计价,计算本条记录的价格
		return d.getPrice(portion) * num;
	}
}

//订单类:保存用户点的所有菜的信息。
class Order {
	Vector<Record> records = new Vector<>();// 保存订单上所有的记录
	int getTotalPrice(String discount) { // 计算订单的总价,discount表示折扣
		double dis = 1;
		if (discount.equals("six")) {
			dis = 0.6;
		}
		if (discount.equals("eight")) {
			dis = 0.8;
		}
		int sum = 0;
		for (Record record:records)
			sum += record.getPrice();
		return (int)Math.round(sum * dis);
	}
	void addASelfRecord(int orderNum, String dishName, int portion, int num,Menu menu) {// 添加一条菜品信息到自己订单中。
		Dish d = menu.searthDish(dishName);
		if (d != null) {  //菜品存在时
			Record record = new Record(orderNum,d,portion,num);
			records.add(record);
			System.out.println(orderNum + " "  +dishName + " " + record.getPrice());
		}
		else {		//菜品不存在时
			Dish dish = new Dish(dishName,0);
			Record record = new Record(orderNum,dish,0,0);
			records.add(record);
			System.out.println(dishName+ " does not exist");
		}
	}
	void addAOtherRecord(int orderNum, String dishName, int portion, int num, Menu menu, Table table, int otherTableNum) {// 添加一条菜品信息到为别人桌订单中。
		Dish d = menu.searthDish(dishName);
		if (d != null) {  //菜品存在时
			Record record = new Record(orderNum,d,portion,num);
			records.add(record);
			System.out.println(orderNum + " table " + table.tableNum + " pay for table " + otherTableNum + " " + record.getPrice());
		}
		else {		//菜品不存在时
			Dish dish = new Dish(dishName,0);
			Record record = new Record(orderNum,dish,0,0);
			records.add(record);
			System.out.println(dishName+ " does not exist");
		}
	}
	void delARecordByOrderNum(int orderNum) {//根据序号删除一条记录
		Record record = findRecordByNum(orderNum);
		if (record == null)
			System.out.println("delete error;");
		else
			records.remove(record);
	}
	Record findRecordByNum(int orderNum) {//根据序号查找一条记录
		for (Record record:records) {
			if (record.orderNum == orderNum)
				return record;
		}
		return null;	
	}
}

class Table {
	Order order;  //订单
	int tableNum;  //桌号
	LocalDate date;  //点餐日期
	int hour;  //点餐时间的小时
	int minute;  //点餐时间的分钟
	Table(int tableNum,LocalDate date,int hour,int minute) {
		this.order = new Order();
		this.tableNum = tableNum;
		this.date = date;
		this.hour = hour;
		this.minute = minute;
	}
	String judgeTime() {
		DayOfWeek dayOfWeek = date.getDayOfWeek();
		if (dayOfWeek.getValue() <= 5 && dayOfWeek.getValue() >= 1) {
			if (hour >= 17 && (hour < 20 || (hour == 20 && minute <= 30)))
				return "eight";  //八折
			else if ((hour > 10 || (hour == 10 && minute >= 30)) && (hour < 14 || (hour == 14 && minute <= 30)))
					return "six";  //六折
		}
		else if (dayOfWeek.getValue() >= 6 && dayOfWeek.getValue() <= 7) {
			if ((hour > 9 || (hour == 9 && minute >= 30)) && (hour < 21 || (hour == 21 && minute <= 30)))
				return "all"; //全价
		}
		return "no"; //不在营业时间内
	}
}

  3、类图

  4、分析

  这次相比2主要多了一个时间的判断,主要考察我们对于java API的运用,对时间做出相应的判断,在不同时间段折扣不同。所以这里的四舍五入顺序也很重要,顺序不对就很可能出现问题,需要注意。

 

  第5次题目集:菜单计价程序-4

  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)//根据序号查找一条记录

}

本次课题比菜单计价系列-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

  2、源码

import java.util.*;

import java.time.Duration;
import java.time.LocalTime;
import java.time.LocalDate;
import java.time.DayOfWeek;
import java.time.format.DateTimeFormatter;

public class Main {
	public static void main(String[] args) {
		Scanner input = new Scanner(System.in);
		Menu menu = new Menu();
		Table[] tables = new Table[100];  //桌数
		int index = -1;  //table数组的序号
		int temp = index;   //用于暂存index
		boolean specialty;  //表示是否为特色菜
		int mixFlag = -1;       //判断菜谱信息与订单信息是否混合,默认为-1为没混合,0为混合
		int orderEffeFlag = -1;    //表示是否为有效的订单,默认为-1表示有效,0表示无效    
		DateTimeFormatter formatter1 = DateTimeFormatter.ofPattern("yyyy/M/d"); // 定义日期时间格式
		DateTimeFormatter formatter2 = DateTimeFormatter.ofPattern("H/m/s");// 定义时间格式
		while (input.hasNextLine()) {  //当有输入时
			String line1 = input.nextLine();
			if (line1.equals("end"))  //遇见end结束
				break;
			String trimmedLine = line1.trim(); // 删除行首和行尾的空格
	        if (!line1.equals(trimmedLine)) {  // 判断删除空格后的字符串和原始字符串是否相等
	        	orderEffeFlag = 0;
	        	System.out.println("wrong format");
	        	continue;
	        }
	        String[] line2 = line1.split(" ");
			if (line2.length == 2) {
				if (orderEffeFlag == -1) {
					if (line2[0].matches("^[0-9]*$") && line2[1].equals("delete")) {  //删除记录
						int deleteNum = Integer.parseInt(line2[0]);
						tables[index].order.delARecordByOrderNum(deleteNum);
					}
					else if (line2[1].matches("^[1-9][0-9]*$")){  //菜单添加菜品信息
						try {
							if (mixFlag == 0)
								throw new DataInputException();	
							specialty = false;
							menu.addDish(line2[0], Integer.parseInt(line2[1]),specialty);
						}
						catch (DataInputException e) {
							System.out.println("invalid dish");
						}
					}
					else
						System.out.println("wrong format");
				}
			}
			else if (line2.length == 3) {  //特色菜
				if (orderEffeFlag == -1) {
					try {
						if (line2[1].matches("^[1-9][0-9]*$") && line2[2].equals("T")) {   //菜单添加菜品信息
							if (mixFlag == 0)
								throw new DataInputException();	
							specialty = true;
							menu.addDish(line2[0], Integer.parseInt(line2[1]),specialty);
						}
						else
							System.out.println("wrong format");
					}
					catch (DataInputException e) {
						System.out.println("invalid dish");
					}
				}
			}
			else if(line2.length == 4) {  
				if (line2[0].equals("table") ) { //新增一桌信息
					int timeFlag = -1;   //判断是时间不合法还是不在营业范围内,默认为-1表示合法,0为时间格式不合法,1为时间数据不合法,2为不在营业范围内
					orderEffeFlag = -1;
					mixFlag = 0;
					try {
						if (!line2[1].matches("^[1-9][0-9]*$")) {
							orderEffeFlag = 0;
							System.out.println("wrong format");
							throw new DataInputException();	
						}
						if (judgeTimeFormatEffective(line2[2],line2[3]) == false) {
							orderEffeFlag = 0;
							timeFlag = 0;
							throw new DataInputException();	
						}
						if (judgeTimeDateEffective(line2[2],line2[3]) == false) {
							orderEffeFlag = 0;
							timeFlag = 1;
							throw new DataInputException();	
						}
						int tableNum = Integer.parseInt(line2[1]);
						LocalDate date = LocalDate.parse(line2[2], formatter1); // 将输入日期的字符串形式转换为刚刚定义的日期时间格式
						LocalTime time = LocalTime.parse(line2[3],formatter2);// 将输入时间的字符串形式转换为刚刚定义的日期时间格式
						Table testTable = new Table(tableNum,date,time);
						String discount = testTable.getDiscount();  //根据时间得到折扣
						if (discount.equals("no")) {
							orderEffeFlag = 0;
							timeFlag = 2;
							throw new DataInputException();	
						}
						if (tableNum > 55) {
							orderEffeFlag = 0;
							System.out.println(tableNum + " table num out of range");
							throw new DataInputException();	
						}
						if (date.getYear() < 2022 || date.getYear() > 2023) {  //时间输入有效但超出范围
							orderEffeFlag = 0;
							System.out.println("not a valid time period");
							throw new DataInputException();	
						}
						System.out.println("table " + tableNum + ": ");
						boolean sameTime = false;  //判断有重复的桌号信息时,两条信息的时间是否在同一时间段,默认为false,表示不在同一时间段
						for (int i=0;i<tables.length;i++) {
							if (tables[i] != null && tableNum == tables[i].tableNum) {
								sameTime = testTable.judgeSameTime(testTable, tables[i]);
								if (sameTime == true) {  //如果重复桌号信息在同一时间段,则后面订单都存于下标为i的桌号信息中,用temp暂存上次桌号信息的下标
									temp = index;
									index = i;
									tables[i].formerOrderNum = -1; //订单序号重新开始比较
								}
							}
						}
						if (sameTime == false) { //不是重复桌号或者重复桌号不在同一时间段,该桌信息存入table数组的下一个
							index = temp;  //获取上次的桌号信息下标
							tables[++index] = testTable;
							temp = index;
						}
					}
					catch (DataInputException e) {
						if (timeFlag == 0)
							System.out.println("wrong format");
						if (timeFlag == 1)
							System.out.println(line2[1] + " date error");
						if (timeFlag == 2)	
							System.out.println("table " + line2[1] + " out of opening hours"); 
					}
				}
				else if (line2[0].matches("^[1-9][0-9]*$") && line2[2].matches("^[1-9]$") && line2[3].matches("^([1-9][0-9]*)$")) {	//自己桌点菜,添加点菜记录
					if (orderEffeFlag == -1) {
						int orderNum = Integer.parseInt(line2[0]);
						String dishName = line2[1];
						int portion = Integer.parseInt(line2[2]);
						int num = Integer.parseInt(line2[3]);
						tables[index].order.addASelfRecord(orderNum, dishName, portion, num, menu,tables[index]);
					}
				}
				else if (orderEffeFlag == -1)
					System.out.println("wrong format");
			}
			else if (line2.length == 5) {  //替其他桌点菜,添加点菜记录
				if (line2[0].matches("^[1-9][0-9]*$") && line2[1].matches("^[1-9][0-9]*$") &&  line2[3].matches("^[1-9]$") && line2[4].matches("^([1-9][0-9]*)$")) {
					if (orderEffeFlag == -1) {
						int otherTableNum=Integer.parseInt(line2[0]);
						int orderNum = Integer.parseInt(line2[1]);
						String dishName = line2[2];
						int portion = Integer.parseInt(line2[3]);
						int num = Integer.parseInt(line2[4]);
						try {
							int tableNumFlag =-1;  //判断代点菜桌号是否存在,默认为-1表示不存在,0表示存在
							for (int i=0;i<index;i++) {
								if (otherTableNum == tables[i].tableNum) {
									tableNumFlag = 0;
									break;
								}
							}
							if (tableNumFlag == -1)
								throw new DataInputException();	
							tables[index].order.addAOtherRecord(orderNum, dishName, portion, num, menu, tables[index],otherTableNum);
						}
						catch (DataInputException e) {
							System.out.println("Table number :" + otherTableNum + " does not exist");
						}
					}
				}
				else if (line2[0].equals("table")) {
					orderEffeFlag = 0;
					System.out.println("wrong format");
				}
				else if (orderEffeFlag == -1)
					System.out.println("wrong format");
			}
			else if (line2[0].equals("table")) {
				orderEffeFlag = 0;
				System.out.println("wrong format");
			}
			else if (orderEffeFlag == -1)
				System.out.println("wrong format");
		}
		for (int i=0;i<=index;i++) {  //计算总价
			for (int n=0;n<tables[index].order.records.size();n++) { //同一桌菜名、份额相同的点菜记录要合并成一条进行计算
				for (int m=0;m<tables[index].order.records.size();m++) {
					String name1 = tables[index].order.records.get(n).d.name;  //第n条记录的菜名
					int portion1 = tables[index].order.records.get(n).portion; //第n条记录的份额
					String name2 = tables[index].order.records.get(m).d.name;  //第m条记录的菜名
					int portion2 = tables[index].order.records.get(m).portion; //第m条记录的份额
					if (n !=m && name1.equals(name2) && portion1 == portion2) {
						tables[index].order.records.get(n).num += tables[index].order.records.get(m).num;
						tables[index].order.records.remove(m);
					}
				}
			}
			String discount = tables[i].getDiscount();  //根据时间得到折扣
			System.out.println("table " + tables[i].tableNum + ": " +tables[i].order.getOriginalTotalPrice() + " " + tables[i].order.getDiscountTotalPrice(discount));
		}
	}
	public static boolean judgeTimeFormatEffective(String date, String times) {
		if (!date.matches("^[0-9]{4}/[0-9]{1,2}/[0-9]{1,2}$"))
			return false;
		if (!times.matches("^[0-9]{1,2}/[0-9]{1,2}/[0-9]{1,2}$"))
			return false;
		return true;
	}
	public static boolean judgeTimeDateEffective(String date,String time) {
		String[] date1 = date.split("/");
		String[] time1 = time.split("/");
		int year = Integer.parseInt(date1[0]); // 获取日期中的年份
		int month = Integer.parseInt(date1[1]); // 获取日期中的月份
		int day = Integer.parseInt(date1[2]); // 获取日期中的日
		int hour = Integer.parseInt(time1[0]); // 获取日期中的年份
		int minute = Integer.parseInt(time1[1]); // 获取日期中的月份
		int second = Integer.parseInt(time1[2]); // 获取日期中的日
		if (month > 12 || month < 1)
			return false;
		if (month == 1 || month == 3 || month == 5 || month == 7 || month == 8 || month == 10 || month == 12) {
			if (day < 1 || day > 31)
				return false;
		}
		if (month == 4 || month == 6 || month == 9 || month == 11) {
			if (day < 1 || day > 30)
				return false;
		}
		if (month == 2) {
			if ((year % 4 == 0 && year % 100 != 0) || year % 400 == 0) {
				if (day < 1 || day > 29)
					return false;
			} else {
				if (day < 1 || day > 28)
					return false;
			}
		}
		if (year == 0 || month == 0 || day == 0)
			return false;
		if (hour < 0 || hour > 24)
			return false;
		if (minute < 0 || minute >= 60)
			return false;
		if (second < 0 || second >= 60)
			return false;
		return true;
	}
}

//菜品类:对应菜谱上一道菜的信息。
class Dish {
	String name;// 菜品名称
	int unit_price; // 单价
	boolean specialty;
	Dish(String name,int unit_price,boolean specialty) {
		this.name = name;
		this.unit_price = unit_price;
		this.specialty = specialty;
	}
	int getPrice(int portion) {// 计算菜品价格的方法,输入参数是点菜的份额(输入数据只能是1/2/3,代表小/中/大份)
		if(portion == 1)
			return unit_price;
		else if (portion == 2)
			return (int)Math.round(unit_price * 1.5);
		else if (portion == 3)
			return unit_price *2;
		else
			return 0;
	}
}

//菜谱类:对应菜谱,包含饭店提供的所有菜的信息。
class Menu {
	Vector<Dish> dishes = new Vector<>();// 菜品数组,保存所有菜品信息
	Dish searthDish(String dishName) { // 根据菜名在菜谱中查找菜品信息,返回Dish对象。
		for (Dish dish:dishes)
			if (dish.name.equals(dishName))
				return dish;
		return null;
	}
	void addDish(String dishName, int unit_price,boolean specialty) { // 添加一道菜品信息
		if (unit_price >= 300) {
			System.out.println(dishName + " price out of range " + unit_price);
			return;
		}
		Dish dish = new Dish(dishName,unit_price,specialty);
		for (int i=0;i<dishes.size();i++) {
			if (dishes.get(i).name.equals(dishName)) {  //如果有多条相同的菜名的记录,菜品的基础价格以最后一条记录为准。
				dishes.set(i, dish);
				return;
			}
		}
		dishes.add(dish);
	}
}

//点菜记录类:保存订单上的一道菜品记录
class Record {
	int orderNum;// 序号
	Dish d;// 菜品
	int portion;// 份额(1/2/3代表小/中/大份)
	int num;  //份数
	Record(int orderNum, Dish d, int portion, int num) {
		this.orderNum = orderNum;
		this.d = d;
		this.portion = portion;
		this.num = num;
	}
	int getPrice() {// 计价,计算本条记录的价格
		return d.getPrice(portion) * num;
	}
}

//订单类:保存用户点的所有菜的信息。
class Order {
	Vector<Record> records = new Vector<>();// 保存订单上所有的记录
	Vector<Integer> deleteNums = new Vector<>();  //保存删除的序号
	int getOriginalTotalPrice () {  //计算订单的原始总价
		int sum = 0;
		for (Record record:records) {
				sum += record.getPrice();				
		}
		return sum;
	}
	int getDiscountTotalPrice(String discount) { // 计算订单的折后总价,discount表示折扣
		double dis = 1;  //周末全价
		if (discount.equals("six")) {
			dis = 0.6;
		}
		if (discount.equals("eight")) {
			dis = 0.8;
		}
		int sum = 0;
		for (Record record:records) {
			if (record.d.specialty == false || dis == 1)
				sum += (int)Math.round(record.getPrice() * dis);
			else
				sum += (int)Math.round(record.getPrice() * 0.7);				
		}
		return sum;
	}
	void addASelfRecord(int orderNum, String dishName, int portion, int num,Menu menu,Table table) {// 添加一条菜品信息到自己订单中。
		Dish d = menu.searthDish(dishName);
		if (orderNum <= table.formerOrderNum) {
			System.out.println("record serial number sequence error"); 
				return;
		}
		if (d== null) {		//菜品不存在时
			Dish dish = new Dish(dishName,0,false);
			Record record = new Record(orderNum,dish,0,0);
			records.add(record);
			System.out.println(dishName+ " does not exist");
			return;
		}
		if (portion > 3 && portion < 10) { //份额超出范围
			System.out.println(orderNum + " portion out of range " + portion);
			return;
		}
		if (num > 15) {  //份数超出范围
			System.out.println(orderNum + " num out of range " + num);
			return;
		}
		table.formerOrderNum = orderNum;
		//菜品存在时
		Record record = new Record(orderNum,d,portion,num);
		records.add(record);
		System.out.println(orderNum + " "  +dishName + " " + record.getPrice());
	}
	void addAOtherRecord(int orderNum, String dishName, int portion, int num, Menu menu, Table table, int otherTableNum) {// 添加一条菜品信息到为别人桌订单中。
		Dish d = menu.searthDish(dishName);
		if (d== null) {		//菜品不存在时
			Dish dish = new Dish(dishName,0,false);
			Record record = new Record(orderNum,dish,0,0);
			records.add(record);
			System.out.println(dishName+ " does not exist");
			return;
		}
		if (portion > 3 && portion < 10) { //份额超出范围
			System.out.println(orderNum + " portion out of range " + portion);
			return;
		}
		if (num > 15) {  //份数超出范围
			System.out.println(orderNum + " num out of range " + num);
			return;
		}
		//菜品存在时
		Record record = new Record(orderNum,d,portion,num);
		records.add(record);
		System.out.println(orderNum + " table " + table.tableNum + " pay for table " + otherTableNum + " " + record.getPrice());
	}
	void delARecordByOrderNum(int orderNum) {//根据序号删除一条记录
		for (int deleteNum:deleteNums) { //判断是否重复删除
			if (orderNum == deleteNum) {
				System.out.println("deduplication " + orderNum);
				return;
			}
		}
		deleteNums.add(orderNum);   //不重复删除就把序号加入动态数组中存储
		Record record = findRecordByNum(orderNum);
		if (record == null)
			System.out.println("delete error;");
		else
			records.remove(record);
	}
	Record findRecordByNum(int orderNum) {//根据序号查找一条记录
		for (Record record:records) {
			if (record.orderNum == orderNum)
				return record;
		}
		return null;	
	}
}

class Table {
	Order order;  //订单
	int tableNum;  //桌号
	LocalDate date;  //点餐日期
	LocalTime time;  //点餐时间
	int hour;  //点餐时间的小时
	int minute;  //点餐时间的分钟
	int formerOrderNum; //用于存储上一个点菜订单记录序号,默认为-1表示还未输入,第一次输入,则直接将订单序号赋值给此变量,往后则用于判断每桌的点菜记录的序号必须按从小到大的顺序排列,若是按从小到大就将新序号赋值给此变量,用于下次继续判断
	Table(int tableNum,LocalDate date,LocalTime time) {
		this.order = new Order();
		this.tableNum = tableNum;
		this.date = date;
		this.time = time;
		this.hour = time.getHour();
		this.minute = time.getMinute();
		this.formerOrderNum = -1;
	}
	String getDiscount() {
		DayOfWeek dayOfWeek = date.getDayOfWeek();
		if (dayOfWeek.getValue() <= 5 && dayOfWeek.getValue() >= 1) {
			if (hour >= 17 && (hour < 20 || (hour == 20 && minute <= 30)))
				return "eight";  //八折
			else if ((hour > 10 || (hour == 10 && minute >= 30)) && (hour < 14 || (hour == 14 && minute <= 30)))
					return "six";  //六折
		}
		else if (dayOfWeek.getValue() >= 6 && dayOfWeek.getValue() <= 7) {
			if ((hour > 9 || (hour == 9 && minute >= 30)) && (hour < 21 || (hour == 21 && minute <= 30)))
				return "all"; //全价
		}
		return "no"; //不在营业时间内
	}
	boolean judgeSameTime(Table table1,Table table2) {
		if (table1.date.equals(table2.date)) {
			if (table1.getDiscount().equals("all") && table2.getDiscount().equals("all")) {
		        Duration duration = Duration.between(table1.time, table2.time); // 计算两个时间之间的时间差
		        long hours = duration.toHours();// 获取时间差的小时数
				if (hours < 1)
					return true;
			}
			else if(table1.getDiscount().equals(table2.getDiscount()))
				return true;
		}
		return false;
	}
}

class DataInputException extends Exception {
	
}

  3、类图

  4、分析

  菜单4的代码明显比之前长了很多,这次主要是增加了很多的非法性判断,主要考察了对于try、catch的用法,什么时候应该throw一个新的错误,然后在对应catch中去处理。也还有很多我直接用的if 判断,所以代码变得长了很多,将近500行了。这次的测试点非常多,有45个,虽然大部分都给出了样例,但是有几个没给出来的真的很难想!而且情况特别多,很容易就漏几个。不是这里少一点,就是那里少了一种情况。卡了我半天,我觉得这谁想得到啊。真的很难想。虽然最后全都过了,可是花了很多时间。

 

 第6次题目集:菜单计价程序-5

  1、题目

本题在菜单计价程序-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

2、源码

import java.util.*;
import java.text.Collator;
import java.time.DayOfWeek;
import java.time.LocalDate;
import java.time.format.DateTimeFormatter;

public class Main {
	public static void main(String[] args) {
		Scanner input = new Scanner(System.in);
		Menu menu = new Menu();
		Vector <Table> tables = new Vector<>(); //桌数
		int index = -1;  //table数组的序号
		boolean specialty;  //表示是否为特色菜,false表示不是特色菜
		int orderEffeFlag = -1;    //表示是否为有效的订单,默认为-1表示有效,0表示无效    
		DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy/M/d"); // 定义日期时间格式
		while (input.hasNextLine()) {  //当有输入时
			String line1 = input.nextLine();
			if (line1.equals("end"))  //遇见end结束
				break;
			String[] line2 = line1.split(" ");
			if (line2.length == 2) {
				if (orderEffeFlag == -1) {
					if (line2[0].matches("^[0-9]*$") && line2[1].equals("delete")) {  //删除记录
						int deleteNum = Integer.parseInt(line2[0]);
						tables.get(index).order.delARecordByOrderNum(deleteNum);
					}
					else if(line2[1].matches("^[1-9][0-9]*$")) {  //菜单添加普通菜品信息
						specialty = false;
						menu.addDish(line2[0],null,Integer.parseInt(line2[1]),specialty);
					}
				}
			}
			else if(line2.length == 4) {  
				if (orderEffeFlag == -1) {
					if (line2[2].matches("^[1-9][0-9]*$") && line2[3].equals("T")) {//菜单添加特色菜品信息
						specialty = true;
						menu.addDish(line2[0],line2[1],Integer.parseInt(line2[2]),specialty);
					}
					else if (line2[0].matches("^[1-9][0-9]*$") && line2[2].matches("^[1-9]$") && line2[3].matches("^([1-9][0-9]*)$")) {	//自己桌点普通菜,添加点菜记录
						int orderNum = Integer.parseInt(line2[0]);
						String dishName = line2[1];
						int portion = Integer.parseInt(line2[2]);
						int num = Integer.parseInt(line2[3]);
						tables.get(index).order.addASelfRecord(orderNum,dishName,-1, portion, num, menu);
					}
				}
			}
			else if (line2.length == 5) {	//自己桌点特色菜,添加点菜记录
				if (orderEffeFlag == -1 && !line2[1].matches("^[1-9][0-9]*$")) {//自己桌点特色菜,添加点菜记录
					int orderNum = Integer.parseInt(line2[0]);
					String dishName = line2[1];
					int tasteLevel = Integer.parseInt(line2[2]);
					int portion = Integer.parseInt(line2[3]);
					int num = Integer.parseInt(line2[4]);
					tables.get(index).order.addASelfRecord(orderNum, dishName,tasteLevel, portion, num, menu);
				}
				if (orderEffeFlag == -1 && line2[1].matches("^[1-9][0-9]*$")) {  //替其他桌点普通菜,添加点菜记录
					int orderNum = Integer.parseInt(line2[1]);
					String dishName = line2[2];
					int portion = Integer.parseInt(line2[3]);
					int num = Integer.parseInt(line2[4]);
					tables.get(index).order.addAOtherRecord(orderNum, dishName, -1,portion, num, menu, tables.get(index), Integer.parseInt(line2[0]),tables);
				}
			}
			else if (line2.length == 6) {  //替其他桌点特色菜,添加点菜记录
				if (orderEffeFlag ==-1) {
					int orderNum = Integer.parseInt(line2[1]);
					String dishName = line2[2];
					int tasteLevel = Integer.parseInt(line2[3]);
					int portion = Integer.parseInt(line2[4]);
					int num = Integer.parseInt(line2[5]);
					tables.get(index).order.addAOtherRecord(orderNum, dishName,tasteLevel, portion, num, menu, tables.get(index), Integer.parseInt(line2[0]),tables);
				}
			}
			else if (line2.length == 7) {
				if (line2[0].equals("table")) { //新增一桌信息
					int tableNum = Integer.parseInt(line2[1]);
					String name = line2[3];
					String tel = line2[4];
					LocalDate date = LocalDate.parse(line2[5], formatter); // 将输入日期的字符串形式转换为刚刚定义的日期时间格式
					String[] time = line2[6].split("/");
					int hour = Integer.parseInt(time[0]);
					int minute = Integer.parseInt(time[1]);
					Table testTable = new Table(tableNum,name,tel,date,hour,minute);
					if (testTable.judgeName() == false || testTable.judgeTel() == false) {
						System.out.println("wrong format");
						orderEffeFlag = 0;
						continue;
					}
					if (testTable.judgeTime().equals("no")) {
						System.out.println("table " + testTable.tableNum + " out of opening hours");
						orderEffeFlag = 0;
						continue;
					}
					System.out.println("table " + tableNum + ": ");
					index++;
					tables.add(testTable);
				}
			}
			else
				System.out.println("wrong format");
		}
		for (int i=0;i<=index;i++) {  //计算总价
			String discount = tables.get(i).judgeTime();  //根据时间得到折扣
			System.out.print("table " + tables.get(i).tableNum + ": " + tables.get(i).order.getOriginalTotalPrice() + " " + tables.get(i).order.getDiscountTotalPrice(discount));
			tables.get(i).order.judgeTasteLevel();
		}
		// 对客户姓名按字母顺序排序
		Collator collator = Collator.getInstance(Locale.CHINA);
		Collections.sort(tables, new Comparator<Table>() {
		    public int compare(Table o1, Table o2) {
		        return collator.compare(o1.name, o2.name);
		    }
		});
		int sum = 0;
		// 输出每位客户的支付金额(按姓名拼音顺序)
		for (int i=0;i<tables.size();i++) {
			String discount = tables.get(i).judgeTime();
			sum += tables.get(i).order.getDiscountTotalPrice(discount);
			if (i == tables.size()-1) {
				System.out.println(tables.get(i).name + " " + tables.get(i).tel + " " + sum);
				break;
			}
			if (!tables.get(i).name.equals(tables.get(i+1).name)) { //如果有相同名字的并在一起算,没有就直接输出
				System.out.println(tables.get(i).name + " " + tables.get(i).tel + " " + sum);
				sum = 0;
			}
		}
	}
}

//菜品类:对应菜谱上一道菜的信息。
class Dish {
	String name;// 菜品名称
	String type;//菜品类型
	int unit_price; // 单价
	boolean specialty;  //菜品是否为特色菜
	Dish(String name,String type,int unit_price,boolean specialty) {
		this.name = name;
		this.type = type;
		this.unit_price = unit_price;
		this.specialty = specialty;
	}
	int getPrice(int portion) {// 计算菜品价格的方法,输入参数是点菜的份额(输入数据只能是1/2/3,代表小/中/大份)
		if(portion == 1)
			return unit_price;
		else if (portion == 2)
			return (int)Math.round(unit_price * 1.5);
		else if (portion == 3)
			return unit_price *2;
		else
			return 0;
	}
}

//菜谱类:对应菜谱,包含饭店提供的所有菜的信息。
class Menu {
	Vector<Dish> dishes = new Vector<>();// 菜品数组,保存所有菜品信息
	Dish searthDish(String dishName) { // 根据菜名在菜谱中查找菜品信息,返回Dish对象。
		for (Dish dish:dishes)
			if (dish.name.equals(dishName))
				return dish;
		return null;
	}
	void addDish(String dishName,String dishType,int unit_price,boolean specialty) { // 添加一道菜品信息
		Dish dish = new Dish(dishName,dishType,unit_price,specialty);
		for (int i=0;i<dishes.size();i++) {
			if (dishes.get(i).name.equals(dishName)) {  //如果有多条相同的菜名的记录,菜品的基础价格以最后一条记录为准。
				dishes.set(i, dish);
				return;
			}
		}
		dishes.add(dish);
	}
}

//点菜记录类:保存订单上的一道菜品记录
class Record {
	int orderNum;// 序号
	Dish d;// 菜品
	int portion;// 份额(1/2/3代表小/中/大份)
	int num;  //份数
	int spicy; //辣度,默认为-1,表示不为特色菜,特色菜在点菜时直接赋值
	int acidity;//酸度,默认为-1,表示不为特色菜,特色菜在点菜时直接赋值
	int sweetness;//甜度,默认为-1,表示不为特色菜,特色菜在点菜时直接赋值
	Record(int orderNum, Dish d, int portion, int num,int spicy,int acidity,int sweetness) {
		this.orderNum = orderNum;
		this.d = d;
		this.portion = portion;
		this.num = num;
		this.spicy = spicy;
		this.acidity = acidity;
		this.sweetness = sweetness;
	}
	int getPrice() {// 计价,计算本条记录的价格
		return d.getPrice(portion) * num;
	}
}

//订单类:保存用户点的所有菜的信息。
class Order {
	Vector<Record> records = new Vector<>();// 保存订单上所有的记录
	int spicySum = -1;  //辣度总和,默认为-1,表示没有这种口味的特色菜
	int aciditySum= -1;	//酸度总和,默认为-1,表示没有这种口味的特色菜
	int sweetnessSum = -1;	//甜度总和,默认为-1,表示没有这种口味的特色菜
	int spicyAve = -1;  //辣度平均数,默认为-1,表示没有这种口味的特色菜
	int acidityAve= -1;	//酸度平均数,默认为-1,表示没有这种口味的特色菜
	int sweetnessAve = -1;	//甜度平均数,默认为-1,表示没有这种口味的特色菜
	int countSpicy = 0; //计算川菜数量
	int countAcidity = 0;//计算晋菜数量
	int countSweetness = 0;//计算浙菜数量
	int getOriginalTotalPrice () {  //计算订单的原始总价
		int sum = 0;
		for (Record record:records) {
				sum += record.getPrice();				
		}
		return sum;
	}
	int getDiscountTotalPrice(String discount) { // 计算订单的折后总价,discount表示折扣
		double dis = 1;  //周末全价
		if (discount.equals("six")) {
			dis = 0.6;
		}
		if (discount.equals("eight")) {
			dis = 0.8;
		}
		int sum = 0;
		for (Record record:records) {
			if (record.d.specialty == false || dis == 1)
				sum += (int)Math.round(record.getPrice() * dis);
			else
				sum += (int)Math.round(record.getPrice() * 0.7);				
		}
		return sum;
	}
	void addASelfRecord(int orderNum, String dishName,int tasteLevel,int portion, int num,Menu menu) {// 添加一条菜品信息到自己订单中。
		Dish d = menu.searthDish(dishName);
		int spicy = -1;
		int acidity = -1;
		int sweetness = -1;
		if (d == null) {		//菜品不存在时
			System.out.println(dishName+ " does not exist");
		}
		else {  //菜品存在时
			if (d.specialty == true) {
				if (d.type.equals("川菜")) {
					if (tasteLevel < 0 || tasteLevel > 5) {
						System.out.println("spicy num out of range :" + tasteLevel);
						return;
					}
					spicy = tasteLevel;
				}
				if (d.type.equals("晋菜")) {
					if (tasteLevel < 0 || tasteLevel > 4) {
						System.out.println("acidity num out of range :" + tasteLevel);
						return;
					}
					acidity = tasteLevel;
				}
				if (d.type.equals("浙菜")) {
					if (tasteLevel < 0 || tasteLevel > 3) {
						System.out.println("sweetness num out of range :" + tasteLevel);
						return;
					}
					sweetness = tasteLevel;
				}
			}
			Record record = new Record(orderNum,d,portion,num,spicy,acidity,sweetness);
			records.add(record);
			System.out.println(orderNum + " "  +dishName + " " + record.getPrice());
		}
	}
	void addAOtherRecord(int orderNum, String dishName,int tasteLevel, int portion, int num, Menu menu, Table table, int otherTableNum,Vector<Table> tables) {// 添加一条菜品信息到为别人桌订单中。
		Dish d = menu.searthDish(dishName);	
		if (d == null) {		//菜品不存在时
			System.out.println(dishName+ " does not exist");
		}
		else {  //菜品存在时
			Record record = new Record(orderNum,d,portion,num,-1,-1,-1);
			records.add(record);
			System.out.println(orderNum + " table " + table.tableNum + " pay for table " + otherTableNum + " " + record.getPrice());
			if (d.specialty == true) {
				for (Table tempTable:tables) {
					if (tempTable.tableNum == otherTableNum) {
						if (d.type.equals("川菜")) {
							if (tasteLevel < 0 || tasteLevel > 5) {
								System.out.println("spicy num out of range :" + tasteLevel);
								return;
							}
							tempTable.order.spicySum += tasteLevel * num;
							tempTable.order.countSpicy += num;
						}
						if (d.type.equals("晋菜")) {
							if (tasteLevel < 0 || tasteLevel > 4) {
								System.out.println("acidity num out of range :" + tasteLevel);
								return;
							}
							tempTable.order.aciditySum += tasteLevel * num;
							tempTable.order.countAcidity += num;
						}
						if (d.type.equals("浙菜")) {
							if (tasteLevel < 0 || tasteLevel > 3) {
								System.out.println("sweetness num out of range :" + tasteLevel);
								return;
							}
							tempTable.order.sweetnessSum += tasteLevel * num;
							tempTable.order.countSweetness += num;
						}
					}
				}
			}
		}
	}
	void delARecordByOrderNum(int orderNum) {//根据序号删除一条记录
		Record record = findRecordByNum(orderNum);
		if (record == null)
			System.out.println("delete error;");
		else
			records.remove(record);
	}
	Record findRecordByNum(int orderNum) {//根据序号查找一条记录
		for (Record record:records) {
			if (record.orderNum == orderNum)
				return record;
		}
		return null;	
	}
	void getAvgTasteLevel() { //计算各种口味度的总值,均值
		for (Record record:records) {
			if (record.spicy != -1) {
				spicySum += record.spicy * record.num;
				countSpicy += record.num;
			}
			if (record.acidity != -1) {
				aciditySum += record.acidity * record.num;
				countAcidity += record.num;
			}
			if (record.sweetness != -1) {
				sweetnessSum += record.sweetness * record.num;
				countSweetness += record.num;
			}
		}
		if (countSpicy > 0) {
			spicySum += 1;
			spicyAve = (int)Math.round((double)spicySum / countSpicy);
		}
		if (countAcidity > 0) {
			aciditySum += 1;
			acidityAve = (int)Math.round((double)aciditySum / countAcidity);
		}
		if (countSweetness > 0) {
			sweetnessSum += 1;
			sweetnessAve = (int)Math.round((double)sweetnessSum / countSweetness);
		}
	}
	void judgeTasteLevel() { //判断是否存在对于特色菜类型,并输出对应信息
		this.getAvgTasteLevel();
		switch (spicyAve) {
		case 0:
			System.out.print(" 川菜 " + countSpicy + " 不辣");
			break;
		case 1:
			System.out.print(" 川菜 " + countSpicy + " 微辣");
			break;
		case 2:
			System.out.print(" 川菜 " + countSpicy + " 稍辣");
			break;
		case 3:
			System.out.print(" 川菜 " + countSpicy + " 辣");
			break;
		case 4:
			System.out.print(" 川菜 " + countSpicy + " 很辣");
			break;
		case 5:
			System.out.print(" 川菜 " + countSpicy + " 爆辣");
			break;
		}
		switch (acidityAve) {
		case 0:
			System.out.print(" 晋菜 " + countAcidity + " 不酸");
			break;
		case 1:
			System.out.print(" 晋菜 " + countAcidity + " 微酸");
			break;
		case 2:
			System.out.print(" 晋菜 " + countAcidity + " 稍酸");
			break;
		case 3:
			System.out.print(" 晋菜 " + countAcidity + " 酸");
			break;
		case 4:
			System.out.print(" 晋菜 " + countAcidity + " 很酸");
			break;
		}
		switch (sweetnessAve) {
		case 0:
			System.out.print(" 浙菜 " + countSweetness + " 不甜");
			break;
		case 1:
			System.out.print(" 浙菜 " + countSweetness + " 微甜");
			break;
		case 2:
			System.out.print(" 浙菜 " + countSweetness + " 稍甜");
			break;
		case 3:
			System.out.print(" 浙菜 " + countSweetness + " 甜");
			break;
		}
		if (countSpicy == 0 && countAcidity == 0 && countSweetness == 0)
			System.out.println(" ");
		else
			System.out.print("\n");
	}
}

class Table {
	Order order;  //订单
	int tableNum;  //桌号
	String name;  //客户姓名
	String tel;  //客户手机号
	LocalDate date;  //点餐日期
	int hour;  //点餐时间的小时
	int minute;  //点餐时间的分钟
	Table(int tableNum,String name,String tel,LocalDate date,int hour,int minute) {
		this.order = new Order();
		this.tableNum = tableNum;
		this.name = name;
		this.tel = tel;
		this.date = date;
		this.hour = hour;
		this.minute = minute;
	}
	boolean judgeName() {  //判断客户姓名是否满足不超过10个字符
		if (name.length() > 10)
			return false;
		return true;
	}
	boolean judgeTel() {  //判断客户手机号11位,前三位必须是180、181、189、133、135、136其中之一。
		if (tel.matches("^1[80|81|89|33|35|36]{2}\\d{8}$"))
			return true;
		return false;
	}
	String judgeTime() {
		DayOfWeek dayOfWeek = date.getDayOfWeek();
		if (dayOfWeek.getValue() <= 5 && dayOfWeek.getValue() >= 1) {
			if (hour >= 17 && (hour < 20 || (hour == 20 && minute <= 30)))
				return "eight";  //八折
			else if ((hour > 10 || (hour == 10 && minute >= 30)) && (hour < 14 || (hour == 14 && minute <= 30)))
					return "six";  //六折
		}
		else if (dayOfWeek.getValue() >= 6 && dayOfWeek.getValue() <= 7) {
			if ((hour > 9 || (hour == 9 && minute >= 30)) && (hour < 21 || (hour == 21 && minute <= 30)))
				return "all"; //全价
		}
		return "no"; //不在营业时间内
	}
}

  3、类图

  4、分析

  这次和4是从3迭代过来的不同分支,但是相比之下,感觉这次的会比4简单一点,4主要是情况太多种了,很容易漏这漏那。这次主要增加的是特色菜和口味度的计算。花的时间也比第四次要少。这次主要是带点菜的情况卡了很久,因为替别人点菜,钱要算在自己桌里面,但是口味度要算在别人那桌,所以一开始一直没有处理好,后来多加了几个变量存储该桌的口味度就解决了。还有就是题目一开始误导了我很久!题目有个地方说输出口味度的平均值,有个地方说输出菜品的数量,我一开始只看见了输出平均值的,一直用平均值算,可是算了很久都算不对,我一直在看给的样例是怎么计算出的结果,发现怎么算都不对,后来又仔细看了一遍题目才发现用的是数量!这里浪费了很久时间,还是自己审题不清的问题。

 

 期中考试:

  期中考试一共有27个选择题,这里就不分析了,大部分都是理论题。四个编程题,主要是最后两个编程题比较重要,主要考了继承和多态。接下来介绍这两题。

  7-3 测验3-继承与多态

  1、题目

将测验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

2、源码

import java.util.*;

public class Main {
    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();
            if (radiums <= 0) {
            	System.out.println("Wrong Format");
            	return;
            	}
            Shape circle = new Circle(radiums);
            printArea(circle);
        case 2://Rectangle
            double x1 = input.nextDouble();
            double y1 = input.nextDouble();
            double x2 = input.nextDouble();
            double y2 = input.nextDouble();
            Point leftTopPoint = new Point(x1,y1);
            Point lowerRightPoint = new Point(x2,y2);
            Rectangle rectangle = new Rectangle(leftTopPoint,lowerRightPoint);
            printArea(rectangle);
            break;
        }
    }
	public static void printArea(Shape shape) {
       System.out.println(String.format("%.2f", shape.getArea()));
    }
}

abstract class Shape {
	abstract double getArea();
}

class Circle extends Shape{
	double radius; //半径
	Circle(double radius) {
		this.radius = radius;
	}
	double getArea() {
		return Math.PI * radius * radius;
	}
}

class Rectangle extends Shape{
	Point topLeftPoint;
	Point lowerRightPoint;
	Rectangle(Point topLeftPoint,Point lowerRightPoint) {
		this.topLeftPoint = topLeftPoint;
		this.lowerRightPoint = lowerRightPoint;
	}
	double getLength() {
		return lowerRightPoint.x - topLeftPoint.x;
	}
	double getHeight() {
		return topLeftPoint.y - lowerRightPoint.y;
	}
	double getArea() {
		return Math.abs(getLength() * getHeight());
	}
}

class Point {
	double x;
	double y;
	Point(double x,double y) {
		this.x = x;
		this.y = y;
	}
}

3、类图

image.png

 4、分析

这题主要考查了继承与多态的用法。继承是指允许一个类继承另一个类的特性(字段和方法)。被继承的类称为父类或超类,继承父类的类称为子类或派生类。通过继承,子类可以获得父类的属性和方法,然后可以扩展或修改它们,使得代码的重用性和可维护性更高。多态是指一个对象可以具有多个类型,并且在不同的情况下可以以不同的方式进行处理。在多态中,父类的引用变量可以引用子类的对象。这使得我们可以编写更通用和灵活的代码。在这里,先声明一个抽象类shape和抽象方法getArea();很明显,当我们使用了抽象类之后,我们不需要分别调用圆和矩形的计算面积方法,只需调用抽象类中的方法即可。然后可以在每个子类中重新修改此方法。

 

  7-4 测验4-抽象类与接口

 1、题目

在测验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 

2、源码

import java.util.*;

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()) + " ");
		}
	}

}

abstract class Shape implements Comparable<Shape> {
	abstract double getArea();

	public int compareTo(Shape other) {
		double thisArea = this.getArea();
		double otherArea = other.getArea();
		return Double.compare(thisArea, otherArea);
	}
}

class Circle extends Shape {
	double radius; // 半径

	Circle(double radius) {
		this.radius = radius;
	}

	double getArea() {
		return Math.PI * radius * radius;
	}
}

class Rectangle extends Shape {
	Point topLeftPoint;
	Point lowerRightPoint;

	Rectangle(Point topLeftPoint, Point lowerRightPoint) {
		this.topLeftPoint = topLeftPoint;
		this.lowerRightPoint = lowerRightPoint;
	}

	double getLength() {
		return lowerRightPoint.x - topLeftPoint.x;
	}

	double getHeight() {
		return topLeftPoint.y - lowerRightPoint.y;
	}

	double getArea() {
		return Math.abs(getLength() * getHeight());
	}
}

class Point {
	double x;
	double y;

	Point(double x, double y) {
		this.x = x;
		this.y = y;
	}
}

 3、分析

  这次主要是考察Shape类要实现Comparable接口和排序的实现。这个知识点感觉老师没讲过,所以当时其实是现学直接套用的,没太搞明白原理。后面会再去了解一下。

三、踩坑心得

  审题一定要仔细!!!有很多测试点过不去,就是因为审题不够仔细,忽略了很多细节,导致耽误了很多时间。还有就是不要偷懒,当时一个四舍五入因为偷懒没用Math.round,导致一直找不到错误,也浪费了很多时间。所以下次开始写代码前一定要认真看几遍题目,然后想好设计哪些类,以及类具体要怎么设计,然后再去开始写。还有就是写代码的时候最好要给一些重要的步骤写上注释,这种不断迭代的,可能后面再看自己前面写的都不记得自己怎么写的,然后重新看一遍,又会浪费许多时间。

四、改进建议

  有时候题目语义不清,或者说的不够清楚,其实会让人有点糊涂,不知道该怎么写,导致后续出现一些问题。建议说清楚一点。

五、总结

  通过这几次作业主要学习了以下内容:

  继承与多态:在 Java 中,通过继承可以让一个类获取另一个类的属性和方法,从而实现代码的重用。子类继承了父类的特性,可以直接使用父类中的方法和字段,同时可以在子类中添加新的方法和字段,或者重写父类中的方法。实现继承的关键字是 extends。多态是指一个对象可以具有多个类型,并且在不同的情况下可以以不同的方式进行处理。在 Java 中,多态通过父类的引用变量指向子类对象来实现。通过调用某个方法,实际执行的是子类中的重写方法。这就是多态的体现,通过父类的引用变量实现对不同子类对象的方法调用,可以提高代码的灵活性和可扩展性。

  异常情况处理:Java提供了异常处理机制,它允许我们在程序运行过程中检测和处理异常情况,从而增加程序的健壮性和可靠性。异常处理可以帮助我们识别和处理潜在的错误或异常情况,以便程序能够正常继续执行,或者在错误发生时进行适当的处理。在Java中,异常是用于表示运行时错误或其他异常情况的对象。异常类以及它们的子类都是从Throwable类派生的。常见的异常类包括Exception和RuntimeException。异常处理的基本结构是try-catch语句,用于捕获和处理可能抛出异常的代码块。try块用于包含可能抛出异常的代码,catch块用于处理异常。如果在try块中的代码引发了异常,那么相应的catch块将被执行。可以有多个catch块来处理不同类型的异常。还可以使用finally块来指定无论异常是否发生都要执行的代码块。除了catch块外,我们还可以使用throw关键字来手动引发异常,以及throws关键字在方法声明中指定可能抛出的异常类型。这允许异常在方法之间传递,并提供了更高级别的异常处理能力。

  以及还有类与类之间的设计方法,代码如何更好的去复用以及java的各种API的正确使用等等。

  这几次不断迭代,题目越来越复杂,学到的知识越来越多,虽然写的时候都很痛苦,但能学的很多新东西,就是之后需要自己花很多时间消化。希望继续努力吧。

标签:空格,记录,int,line2,BLOG,菜品,table
From: https://www.cnblogs.com/Blue-zzz/p/17842828.html

相关文章

  • BLOG-2
    一、前言1.知识点运用:子类父类的继承关系,以及各种类之间的依赖、关联、聚合和组合等关系。2.题量:菜单4,菜单5以及期中考试的三道题目。3.难度:菜单4和菜单5较为困难,跟菜单3相比,菜单4增加了许多错误情况,而菜单5则是在特色菜和订多桌菜方面进行了迭代。二、设计与分析7-1菜单计......
  • BLOG-2
    期中考试的题目相对比较简单但是具有典型意义,而菜单系列的题目难度逐渐增加,总体来说需要认真对待。7-1菜单计价程序-3分数40作者 蔡轲单位 南昌航空大学设计点菜计价程序,根据输入的信息,计算并输出总价格。输入内容按先后顺序包括两部分:菜单、订单,最后......
  • blog2
    一、题目:7-1菜单计价程序-4本体大部分内容与菜单计价程序-3相同,增加的部分用加粗文字进行了标注。设计点菜计价程序,根据输入的信息,计算并输出总价格。输入内容按先后顺序包括两部分:菜单、订单,最后以"end"结束。菜单由一条或多条菜品记录组成,每条记录一行每条菜品记录包含:菜......
  • Blog2
    前言此次博客是关于菜单计价程序-4、5及期中测试的一次总结。闲话少说,先看题目。设计与分析7-1菜单计价程序-4分数100全屏浏览题目切换布局作者 蔡轲单位 南昌航空大学本体大部分内容与菜单计价程序-3相同,增加的部分用加粗文字进行了标注。......
  • 小结blog2
    一、前言这次总结是对第四次题目集,第五次题目集,第六次题目集和期中考试的总结1.第四次题目集主要是菜单三简单构建,实现复杂任务类的设计2.第五次题目集主是:菜单计价程序4,在基于菜单计价程序3的基础上进行完善,增加了新的需求,主要是异常情况3.第六次题目集主是:......
  • 21207328-吴义隆-Blog2
    一、前言:(1)知识点本次作业,知识点很广,单论期中考试,就涉及了许多,类结构设计、继承与多态和抽象类与接口等。Java类结构知识点:类的声明:Java类通过关键字"class"来声明,后面跟着类名和类体,类体包含类的成员变量和方法。成员变量:也称为字段或属性,用于描述类的状态。它们定......
  • PTA4-6、期中考试总结Blog-2
    本次Blog是针对Java程序设计的PTA题目集4-6以及期中考试的总结。一、前言:题目集4共四道题,包括菜单计价系列题目两道、单词统计与排序、 判断两个日期的先后,计算间隔天数、周数。题目难度适中,其中菜单计价2-3难度较大。考察的主要是正则表达式运算、哈希表的运用等。题目集5只有......
  • BLOG-2
    BLOG-2前言本次blog主要是对PTA题目集4、5、6以及期中考试的总结及分析,重点是对菜单计价四、五以及其中考试的总结。菜单计价四和五都是在菜单计价三的基础上的延伸,菜单计价四增加了很多错误输入情况,菜单计价五则是在三的条件下去延伸了菜的一些特色。这两次的大作业和期中......
  • BLOG-2
    importjava.time.DayOfWeek;importjava.time.LocalDate;importjava.time.LocalDateTime;importjava.time.LocalTime;importjava.time.format.DateTimeFormatter;importjava.util.*;classUserInformationimplementsComparable<UserInformation>{publicSt......
  • java-blog2
    一、前言1、知识点:主要是java中类与类之间的关系和类图的构建以及对数据的处理2、题量:题量不大,简单的作业题量多点,难点的作业题量少点总体来说还行3、难度:难度在不断的变大,每一题所花的时间也在不断的增加二、设计与分析:第四次大作业:设计点菜计价程序,根据输入的信息,计算并输......