pta第三次blog作业
- 前言
学期的尾声,学的东西也越来越多,这几次pta有些题目不是单一的要实现某个功能,而是要设计系统,包含许多功能还有逻辑问题,比之前难一点吧。题量没有太大变化,这几次pta主要是程序的迭代,迭代的题目第一次有一个基础之后做后面的题就不会很费劲了,除了迭代的题,其他的难度都是不很大。知识点有几种容器,set,hashmap,treemap。还有输入输出,arraylist,接口,继承,还有栈,除了set和map不太熟悉之外,其他在以前的题目中都有涉及到。
- 试题分析
题目集07:
本题在菜单计价程序-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"
参考以下类的模板进行设计:菜品类:对应菜谱上一道菜的信息。
### 输入格式:
桌号标识格式: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
设计与分析:
这题是之前pta题目的迭代,主要难度在 Main函数里对输入信息的读入,处理和输出,这道题输入的信息的错误类型很多,就要分很多种情况来处理,需要考虑其中的逻辑问题,拿不到分主要就是没有考虑周全以及处理错误类型的逻辑问题。这道题读入信息采用的是一行一行读,对每行都做一个格式判断,然后以空格分开做进一步处理,对错误类型的处理穿插在其中。这题多加了table和uesr类,table类型的对象数组作为user类的属性,方便最后统计和输出。
类图:

SourceMonitor的生成报表内容:

采坑心得:
这题主要就是逻辑还有考虑不周全的问题,要多用调试功能就能较为快速地找到问题所在,还有就是自己要多设置尝试不同的测试点。
改进建议:
程序比较难读,对于许多问题的处理不够恰当高效,要多实现单一职责原则。把功能细分。
源码如下:
查看代码 import java.util.ArrayList;
import java.util.Scanner;
public class Main {
public static void main(String[] args) {
// TODO Auto-generated method stub
Scanner input = new Scanner(System.in);
String menuMessage = input.nextLine();
Menu menu = new Menu();
int j = 0;
while (!menuMessage.startsWith("table")) {
String string[] = menuMessage.split(" ");
if (checkDishValidity(menuMessage)) {// 基础格式
if (menu.searthDish(string[0]) == null) {// 没有找到相同名称的菜
if (string.length > 2) {// 特色菜
menu.addDish(string[0], string[1], Integer.parseInt(string[2]));
} else {// 非特色菜
menu.addDish(string[0], null, Integer.parseInt(string[1]));
}
} else {// 找到了相同名称的菜
if (string.length == 4) {// 找到的是特色菜
menu.searthDish(string[0]).unit_price = Integer.parseInt(string[2]);
} else {
menu.searthDish(string[0]).unit_price = Integer.parseInt(string[1]);
}
}
} else {// 非法输入
System.out.println("wrong format");
}
menuMessage = input.nextLine();
j++;
}
String tableMessage = menuMessage;// 上述循环会多读一条
ArrayList<Table> tables = new ArrayList<Table>();
ArrayList<User> users = new ArrayList<User>();
while (true) {
Order order = new Order();// 开始读订单信息
if (checkTableValidity(tableMessage)) {// 基础格式正确
String[] string = tableMessage.split(" ");
String userName = string[3];
String phoneNum = string[4];
String[] Time1 = string[5].split("/");
String[] Time2 = string[6].split("/");
int day = Integer.parseInt(Time1[2]);
int month = Integer.parseInt(Time1[1]);
int year = Integer.parseInt(Time1[0]);
int hour = Integer.parseInt(Time2[0]);
int minute = Integer.parseInt(Time2[1]);
int second = Integer.parseInt(Time2[2]);
int tableNum = Integer.parseInt(string[1]);
if (checkTimeValidity(year, month, day, hour, minute, second)) {// 时间是否合法
if (isBusinessTime(year, month, day, hour, minute, second)) {// 在营业时间
String recordMessage = input.nextLine();
while (true) {// 读入各桌菜品记录
if (recordMessage.startsWith("table") || recordMessage.startsWith("end")) {
break;
}
String string1[] = recordMessage.split(" ");
String dname = string1[1];
if (checkRecordValidity(recordMessage) == 2) {/////////////// 格式正确,非特色菜格式
String number = string1[0];
int portion = Integer.parseInt(string1[2]);
int copies = Integer.parseInt(string1[3]);
if (menu.searthDish(dname) == null) {
System.out.println(dname + " does not exist");
} else {
if (menu.searthDish(dname).taste == null) {
if (portion > 3) {
portion = 3;
}
Record record = new Record(number, menu.searthDish(dname), 0, portion, copies);
order.addARecord(record);
} else {
System.out.println("wrong format");
}
}
} else if (checkRecordValidity(recordMessage) == 1) {////////////////////// 格式正确,特色菜格式
String number = string1[0];
int portion = Integer.parseInt(string1[3]);
int copies = Integer.parseInt(string1[4]);
int degree = Integer.parseInt(string1[2]);
if (menu.searthDish(dname) == null) {
System.out.println(dname + " does not exist");
} else {
if (menu.searthDish(dname).taste == null) {
System.out.println("wrong format");
} else {
if (portion > 3) {
portion = 3;
}
Record record = new Record(number, menu.searthDish(dname), degree, portion,
copies);
order.addARecord(record);
}
}
} else if (checkDeleteValidity(recordMessage)) {// 删除基础格式正确
String[] deleteMessage = recordMessage.split(" ");
if (!order.findRecordByNum(deleteMessage[0])) {
System.out.println("delete error");
} else {
order.delARecordByOrderNum(deleteMessage[0]);
}
} else {
System.out.println("wrong format");
}
recordMessage = input.nextLine();
tableMessage = recordMessage;/*
* 如果是当前订单最后一条菜品记录,但已经读了下一条,
* 所以此时让tableMessage=recordMessage,就算不是循环也不会停
*/
}
Table table = new Table(tableNum, order);
table.time[0] = year;
table.time[1] = month;
table.time[2] = day;
table.time[3] = hour;
table.time[4] = minute;
table.time[5] = second;
tables.add(table);
int flag = 0;
if (users.isEmpty()) {
User user = new User();
user.setName(userName);
user.setPhonenum(phoneNum);
user.tables.add(table);
users.add(user);
} else {
for (int k = 0; k < users.size(); k++) {
if (users.get(k).name.equals(userName)) {// 一人多桌
users.get(k).tables.add(table);
flag = 1;
}
}
if (flag == 0) {
User user = new User();
user.setName(userName);
user.setPhonenum(phoneNum);
user.tables.add(table);
users.add(user);
}
}
} else {
System.out.println("table " + tableNum + " out of opening hours");
tableMessage = input.nextLine();
while (true) {
if (tableMessage.startsWith("table") || tableMessage.startsWith("end")) {
break;
}
tableMessage = input.nextLine();
}
}
} else {// 非法输入
System.out.println("wrong format");
tableMessage = input.nextLine();
while (true) {
if (tableMessage.startsWith("table") || tableMessage.startsWith("end")) {
break;
}
tableMessage = input.nextLine();
}
}
} else {// 非法输入
System.out.println("wrong format");
tableMessage = input.nextLine();
while (true) {
if (tableMessage.startsWith("table") || tableMessage.startsWith("end")) {
break;
}
tableMessage = input.nextLine();
}
}
if (tableMessage.startsWith("end")) {
break;
}
}
for (int k = 0; k < tables.size(); k++) { // 输出桌信息
System.out.println("table " + tables.get(k).tableNum + ": ");
for (int l = 0; l < tables.get(k).order.records.size(); l++) {// 输出对应桌的菜品信息
if (tables.get(k).order.records.get(l).degree > 5
&& tables.get(k).order.records.get(l).d.taste.equals("川菜")) {
System.out.println("spicy num out of range :" + tables.get(k).order.records.get(l).degree);
} else if (tables.get(k).order.records.get(l).degree > 4
&& tables.get(k).order.records.get(l).d.taste.equals("晋菜")) {
System.out.println("acidity num out of range :" + tables.get(k).order.records.get(l).degree);
} else if (tables.get(k).order.records.get(l).degree > 3
&& tables.get(k).order.records.get(l).d.taste.equals("浙菜")) {
System.out.println("sweetness num out of range :" + tables.get(k).order.records.get(l).degree);
} else {
System.out.print(tables.get(k).order.records.get(l).orderNum + " "
+ tables.get(k).order.records.get(l).d.name + " ");
System.out.println(Math.round(tables.get(k).order.records.get(l).getPrice1()));// 输出每条记录的菜价
}
}
}
for (int k = 0; k < tables.size(); k++) {// 输出对应桌的菜品信息
System.out.print("table " + tables.get(k).tableNum + ": ");
System.out.print(tables.get(k).getPrice1() + " ");// 输出每桌原价
System.out.print(tables.get(k).getPrice2());// 输出每桌总价
if (tables.get(k).isContainSichuancuisine()) {
if (tables.get(k).isContainJincuisine() || tables.get(k).isContainZhejiangcuisine()) {
System.out.print(
" 川菜 " + tables.get(k).getTotallcopies("川菜") + " " + tables.get(k).getFinalDegree("川菜"));
} else
System.out.println(
" 川菜 " + tables.get(k).getTotallcopies("川菜") + " " + tables.get(k).getFinalDegree("川菜"));
}
if (tables.get(k).isContainJincuisine()) {
if (tables.get(k).isContainZhejiangcuisine()) {
System.out.print(
" 晋菜 " + tables.get(k).getTotallcopies("晋菜") + " " + tables.get(k).getFinalDegree("晋菜"));
} else
System.out.println(
" 晋菜 " + tables.get(k).getTotallcopies("晋菜") + " " + tables.get(k).getFinalDegree("晋菜"));
}
if (tables.get(k).isContainZhejiangcuisine()) {
System.out.println(
" 浙菜 " + tables.get(k).getTotallcopies("浙菜") + " " + tables.get(k).getFinalDegree("浙菜"));
}
}
for (int m = 0; m < users.size(); m++) {// 排列用户信息
for (int n = 0; n < users.size() - m - 1; n++) {
for (int p = 0; p < users.get(n).name.length(); p++) {
if (users.get(n).name.toLowerCase().charAt(p) > users.get(n + 1).name.toLowerCase().charAt(p)) {
User temp = new User();
temp = users.get(n);
users.set(n, users.get(n + 1));
users.set(n + 1, temp);
break;
}
if (users.get(n).name.toLowerCase().charAt(p) < users.get(n + 1).name.toLowerCase().charAt(p))
break;
}
}
}
for (int m = 0; m < users.size(); m++) {// 输出用户信息
System.out.println(users.get(m).name + " " + users.get(m).phonenum + " " + users.get(m).getPrice());
}
}
public static boolean isBusinessTime(int year, int month, int day, int hour, int minute, int second) {
int week = (day + 2 * month + 3 * (month + 1) / 5 + year + year / 4 - year / 100 + year / 400 + 1) % 7;
if (week >= 1 && week <= 5) {// 周一到周五
if (hour > 10 && hour < 14 || (hour == 10 && minute >= 30) || (hour == 14 && minute <= 30)) {// 中午
return true;
} else if (hour >= 17 && hour < 20 || (hour == 20 && minute <= 30)) {// 晚上
return true;
} else {// 不在周一至周五的营业时间
return false;
}
} else {// 周末
if (hour > 9 && hour < 21 || (hour == 9 && minute >= 30) || (hour == 21 && minute <= 30)) {
return true;
} else {// 不在周末的营业时间
return false;
}
}
}
public static boolean checkTimeValidity(int year, int month, int day, int hour, int minute, int second) {
int[] mon_maxnum = { 0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 };
if (year >= 1000 && month >= 1 && month <= 12 && day >= 1) {
if ((year % 400 == 0 || year % 4 == 0 && year % 100 != 0) && month == 2) {
if (day <= 29 && hour <= 24 && hour >= 0 && minute >= 0 && minute <= 60 && second >= 0
&& second <= 60) {
return true;
} else {
return false;
}
} else if (day <= mon_maxnum[month] && hour <= 24 && hour >= 0 && minute >= 0 && minute <= 60 && second >= 0
&& second <= 60)
return true;
else
return false;
} else
return false;
}
public static boolean checkDishValidity(String menuMessage) {
if (menuMessage.matches(".+\\s[1-9][\\d]{0,}+")) {
return true;
}
if (menuMessage.matches(".+\\s.+\\s[1-9][\\d]{0,}\\sT")) {
return true;
}
return false;
}
public static boolean checkTableValidity(String tableMessage) {
if (tableMessage.matches(
"table\\s[\\d]+\\s:\\s.{1,11}\\s(180|181|189|133|135|136)[\\d]{8}\\s[\\d]+/[\\d]+/[\\d]+\\s[\\d]+/[\\d]+/[\\d]+")) {
return true; // table+英文空格+桌号+英文空格+":"+英文空格+客户姓名+英文空格+手机号+日期(格式:YYYY/MM/DD)+英文空格+ 时间(24小时制格式:
// HH/MM/SS)
} else {
return false;
}
}
public static int checkRecordValidity(String recordMessage) {
if (recordMessage.matches("[\\d]\\s.+\\s[\\d]+\\s[\\d]+\\s[\\d]+")) {
return 1;// 序号+英文空格+菜名+英文空格+口味度值+英文空格+份额+英文空格+份数
}
if (recordMessage.matches("[\\d]\\s.+\\s[\\d]+\\s[\\d]+")) {
return 2;
}
return 0;
}
public static boolean checkDeleteValidity(String recordMessage) {
if (recordMessage.matches("[\\d]+\\sdelete")) {
return true;
} else {
return false;
}
}
}
class Dish {
String name;// 菜品名称
int unit_price; // 单价
String taste;
public Dish() {
super();
// TODO Auto-generated constructor stub
}
public Dish(String name, int unit_price, String taste) {
super();
this.name = name;
this.unit_price = unit_price;
this.taste = taste;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getUnit_price() {
return unit_price;
}
public void setUnit_price(int unit_price) {
this.unit_price = unit_price;
}
public int getPrice(int portion) {// 计算菜品价格的方法,输入参数是点菜的份额(输入数据只能是1/2/3,代表小/中/大份)
return (int) Math.round(unit_price * (1 + (portion - 1) * 0.5));
}
}
class Menu {
ArrayList<Dish> dishs = new ArrayList<Dish>();// 菜品数组,保存所有菜品信息
public ArrayList<Dish> getDishs() {
return dishs;
}
public void setDishs(ArrayList<Dish> dishs) {
this.dishs = dishs;
}
public Menu(ArrayList<Dish> dishs) {
super();
this.dishs = dishs;
}
public Menu() {
super();
// TODO Auto-generated constructor stub
}
public Dish searthDish(String dishName) {// 根据菜名在菜谱中查找菜品信息,返回Dish对象。
if (dishs == null)
return null;
for (int i = 0; i < dishs.size(); i++) {
if (dishName.equals(dishs.get(i).name)) {
return dishs.get(i);
}
}
return null;
}
public Dish addDish(String dishName, String taste, int unit_price) {// 添加一道菜品信息
Dish dish = new Dish(dishName, unit_price, taste);
dishs.add(dish);
return dish;
}
}
class Record {
String orderNum;// 序号
Dish d;// 菜品
int degree;// 口味度
int portion;// 份额(1/2/3代表小/中/大份)
int amout;// 份数
boolean state = true;
public String getOrderNum() {
return orderNum;
}
public void setOrderNum(String orderNum) {
this.orderNum = orderNum;
}
public Dish getD() {
return d;
}
public void setD(Dish d) {
this.d = d;
}
public int getPortion() {
return portion;
}
public void setPortion(int portion) {
this.portion = portion;
}
public Record(String orderNum, Dish d, int degree, int portion, int amout) {
super();
this.orderNum = orderNum;
this.d = d;
this.degree = degree;
this.portion = portion;
this.amout = amout;
}
public Record(String orderNum, Dish d, int portion, int amout) {
super();
this.orderNum = orderNum;
this.d = d;
this.portion = portion;
this.amout = amout;
}
public Record() {
super();
// TODO Auto-generated constructor stub
}
public int getPrice() {// 计价,计算本条记录的价格
if ((degree > 5 && d.taste.equals("川菜")) || (degree > 4 && d.taste.equals("晋菜"))
|| (degree > 3 && d.taste.equals("浙菜")) || (state == false)) {
return 0;
}
return d.getPrice(portion) * amout;
}
public int getPrice1() {// 计价,计算本条记录的价格
if ((degree > 5 && d.taste.equals("川菜")) || (degree > 4 && d.taste.equals("晋菜"))
|| (degree > 3 && d.taste.equals("浙菜"))) {
return 0;
}
return d.getPrice(portion) * amout;
}
}
class Order {
ArrayList<Record> records = new ArrayList<Record>();// 保存订单上每一道的记录
public ArrayList<Record> getRecords() {
return records;
}
public void setRecords(ArrayList<Record> records) {
this.records = records;
}
public Order(ArrayList<Record> records) {
super();
this.records = records;
}
public Order() {
super();
// TODO Auto-generated constructor stub
}
public void addARecord(Record record) {// 添加一条菜品信息到订单中。
records.add(record);
}
public void delARecordByOrderNum(String orderNum) {// 根据序号删除一条记录
for (int i = 0; i < records.size(); i++) {
if (records.get(i).orderNum.equals(orderNum)) {
records.get(i).state = !records.get(i).state;
}
}
}
public boolean findRecordByNum(String orderNum) {// 根据序号查找一条记录
for (int i = 0; i < records.size(); i++) {
if (records.get(i).orderNum.equals(orderNum)) {
return true;
}
}
return false;
}
}
class Table {
int tableNum;
Order order;
int[] time = new int[6];
public Table() {
super();
// TODO Auto-generated constructor stub
}
public int getTableNum() {
return tableNum;
}
public void setTableNum(int tableNum) {
this.tableNum = tableNum;
}
public Order getOrder() {
return order;
}
public void setOrder(Order order) {
this.order = order;
}
public Table(int tableNum, Order order) {
super();
this.tableNum = tableNum;
this.order = order;
}
public long getPrice2() {
long totallPrice = 0;
for (int i = 0; i < order.records.size(); i++) {
int year = time[0];
int month = time[1];
int day = time[2];
int hour = time[3];
int minute = time[4];
int second = time[5];
int week = (day + 2 * month + 3 * (month + 1) / 5 + year + year / 4 - year / 100 + year / 400 + 1) % 7;
if (week >= 1 && week <= 5 && order.records.get(i).d.taste != null) {
totallPrice += Math.round(order.records.get(i).getPrice() * 0.7);
} else if (week >= 1 && week <= 5 && order.records.get(i).d.taste == null && hour >= 17 && hour < 20
|| (hour == 20 && minute <= 30)) {
totallPrice += Math.round(order.records.get(i).getPrice() * 0.8);
} else if (week >= 1 && week <= 5 && order.records.get(i).d.taste == null && hour > 10 && hour < 14
|| (hour == 10 && minute >= 30) || (hour == 14 && minute <= 30)) {
totallPrice += Math.round(order.records.get(i).getPrice() * 0.6);
} else {
totallPrice += Math.round(order.records.get(i).getPrice());
}
}
return totallPrice;
}
public int getPrice1() {
int totallPrice = 0;
for (int i = 0; i < order.records.size(); i++) {
totallPrice += Math.round(order.records.get(i).getPrice());
}
return totallPrice;
}
public boolean isContainSichuancuisine() {
for (int i = 0; i < order.records.size(); i++) {
if (order.records.get(i).d.taste != null) {
if (order.records.get(i).d.taste.equals("川菜") && order.records.get(i).state == true) {
return true;
}
}
}
return false;
}
public boolean isContainZhejiangcuisine() {
for (int i = 0; i < order.records.size(); i++) {
if (order.records.get(i).d.taste != null) {
if (order.records.get(i).d.taste.equals("浙菜") && order.records.get(i).state == true) {
return true;
}
}
}
return false;
}
public boolean isContainJincuisine() {
for (int i = 0; i < order.records.size(); i++) {
if (order.records.get(i).d.taste != null) {
if (order.records.get(i).d.taste.equals("晋菜") && order.records.get(i).state == true) {
return true;
}
}
}
return false;
}
public int getAverageDegree(String taste) {
double totall = 0;
double num = 0;
for (int i = 0; i < order.records.size(); i++) {
if (order.records.get(i).d.taste != null) {
if (order.records.get(i).d.taste.equals(taste)) {
if ((order.records.get(i).degree > 5 && order.records.get(i).d.taste.equals("川菜"))
|| (order.records.get(i).degree > 4 && order.records.get(i).d.taste.equals("晋菜"))
|| (order.records.get(i).degree > 3 && order.records.get(i).d.taste.equals("浙菜"))
|| (order.records.get(i).state == false)) {
;
} else {
totall += order.records.get(i).degree * order.records.get(i).amout;
num += order.records.get(i).amout;
}
}
}
}
return (int) Math.round(totall / num);
}
public int getTotallcopies(String taste) {
int totall = 0;
for (int i = 0; i < order.records.size(); i++) {
if (order.records.get(i).d.taste != null) {
if (order.records.get(i).d.taste.equals(taste)) {
if ((order.records.get(i).degree > 5 && order.records.get(i).d.taste.equals("川菜"))
|| (order.records.get(i).degree > 4 && order.records.get(i).d.taste.equals("晋菜"))
|| (order.records.get(i).degree > 3 && order.records.get(i).d.taste.equals("浙菜"))
|| (order.records.get(i).state == false)) {
;
} else {
totall += order.records.get(i).amout;
}
}
}
}
return Math.round(totall);
}
public String getFinalDegree(String taste) {
if (taste.equals("川菜")) {
if (getAverageDegree("川菜") == 0) {// 不辣、微辣、稍辣、辣、很辣、爆辣
return "不辣";
} else if (getAverageDegree("川菜") == 1) {
return "微辣";
} else if (getAverageDegree("川菜") == 2) {
return "稍辣";
} else if (getAverageDegree("川菜") == 3) {
return "辣";
} else if (getAverageDegree("川菜") == 4) {
return "很辣";
} else {
return "爆辣";
}
}
if (taste.equals("晋菜")) {// 不酸、微酸、稍酸、酸、很酸
if (getAverageDegree("晋菜") == 0) {
return "不酸";
} else if (getAverageDegree("晋菜") == 1) {
return "微酸";
} else if (getAverageDegree("晋菜") == 2) {
return "稍酸";
} else if (getAverageDegree("晋菜") == 3) {
return "酸";
} else {
return "很酸";
}
}
if (taste.equals("浙菜")) {// 不甜、微甜、稍甜、甜
if (getAverageDegree("浙菜") == 0) {
return "不甜";
} else if (getAverageDegree("浙菜") == 1) {
return "微甜";
} else if (getAverageDegree("浙菜") == 2) {
return "稍甜";
} else {
return "甜";
}
}
return null;
}
}
class User {
String name;
String phonenum;
ArrayList<Table> tables = new ArrayList<Table>();
public User() {
super();
// TODO 自动生成的构造函数存根
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getPhonenum() {
return phonenum;
}
public void setPhonenum(String phonenum) {
this.phonenum = phonenum;
}
public ArrayList<Table> getTables() {
return tables;
}
public void setTables(ArrayList<Table> tables) {
this.tables = tables;
}
public User(String name, String phonenum, ArrayList<Table> tables) {
super();
this.name = name;
this.phonenum = phonenum;
this.tables = tables;
}
public int getPrice() {
int totallPrice = 0;
for (int i = 0; i < tables.size(); i++) {
totallPrice += tables.get(i).getPrice2();
}
return totallPrice;
}
}
题目集08:
某高校课程从性质上分为:必修课、选修课,从考核方式上分为:考试、考察。
考试的总成绩由平时成绩、期末成绩分别乘以权重值得出,比如平时成绩权重0.3,期末成绩权重0.7,总成绩=平时成绩*0.3+期末成绩*0.7。
考察的总成绩直接等于期末成绩
必修课的考核方式必须为考试,选修课可以选择考试、考察任一考核方式。
1、输入:
包括课程、课程成绩两类信息。
课程信息包括:课程名称、课程性质、考核方式(可选,如果性质是必修课,考核方式可以没有)三个数据项。
课程信息格式:课程名称+英文空格+课程性质+英文空格+考核方式
课程性质输入项:必修、选修
考核方式输入选项:考试、考察
课程成绩信息包括:学号、姓名、课程名称、平时成绩(可选)、期末成绩
课程信息格式:学号+英文空格+姓名+英文空格+课程名称+英文空格+平时成绩+英文空格+期末成绩
以上信息的相关约束:
1)平时成绩和期末成绩的权重默认为0.3、0.7
2)成绩是整数,不包含小数部分,成绩的取值范围是【0,100】
3)学号由8位数字组成
4)姓名不超过10个字符
5)课程名称不超过10个字符
6)不特别输入班级信息,班级号是学号的前6位。
2、输出:
输出包含三个部分,包括学生所有课程总成绩的平均分、单门课程成绩平均分、单门课程总成绩平均分、班级所有课程总成绩平均分。
为避免误差,平均分的计算方法为累加所有符合条件的单个成绩,最后除以总数。
1)学生课程总成绩平均分按学号由低到高排序输出
格式:学号+英文空格+姓名+英文空格+总成绩平均分
如果某个学生没有任何成绩信息,输出:学号+英文空格+姓名+英文空格+"did not take any exams"
2)单门课程成绩平均分分为三个分值:平时成绩平均分(可选)、期末考试平均分、总成绩平均分,按课程名称的字符顺序输出
格式:课程名称+英文空格+平时成绩平均分+英文空格+期末考试平均分+英文空格+总成绩平均分
如果某门课程没有任何成绩信息,输出:课程名称+英文空格+"has no grades yet"
3)班级所有课程总成绩平均分按班级由低到高排序输出
格式:班级号+英文空格+总成绩平均分
如果某个班级没有任何成绩信息,输出:班级名称+英文空格+ "has no grades yet"
异常情况:
1)如果解析某个成绩信息时,课程名称不在已输入的课程列表中,输出:学号+英文空格+姓名+英文空格+":"+课程名称+英文空格+"does not exist"
2)如果解析某个成绩信息时,输入的成绩数量和课程的考核方式不匹配,输出:学号+英文空格+姓名+英文空格+": access mode mismatch"
以上两种情况如果同时出现,按第一种情况输出结果。
3)如果解析某个课程信息时,输入的课程性质和课程的考核方式不匹配,输出:课程名称+" : course type & access mode mismatch"
4)格式错误以及其他信息异常如成绩超出范围等,均按格式错误处理,输出"wrong format"
5)若出现重复的课程/成绩信息,只保留第一个课程信息,忽略后面输入的。
信息约束:
1)成绩平均分只取整数部分,小数部分丢弃
参考类图:

输入样例1:
仅有课程。例如:
查看代码 java 必修 考试
数据结构 选修 考试
形式与政治 选修 考察
end
输出样例1:
在这里给出相应的输出。例如:
查看代码 java has no grades yet
数据结构 has no grades yet
形式与政治 has no grades yet
输入样例2:
单门考试课程 单个学生。例如:
查看代码 java 必修 考试
20201103 张三 java 20 40
end
输出样例2:
在这里给出相应的输出。例如:
查看代码 20201103 张三 34
java 20 40 34
202011 34
输入样例3:
单门考察课程 单个学生。例如:
查看代码 java 选修 考察
20201103 张三 java 40
end
输出样例3:
在这里给出相应的输出。例如:
查看代码 20201103 张三 40
java 40 40
202011 40
输入样例4:
考试课程 单个学生 不匹配的考核方式。例如:
查看代码 java 必修 考试
20201103 张三 java 20
end
输出样例4:
在这里给出相应的输出。例如:
查看代码 20201103 张三 : access mode mismatch
20201103 张三 did not take any exams
java has no grades yet
202011 has no grades yet
输入样例5:
单门课程,单个学生,课程类型与考核类型不匹配。例如:
查看代码 java 必修 考察
20201103 张三 java 40
end
输出样例5:
在这里给出相应的输出。例如:
查看代码 java : course type & access mode mismatch
java does not exist
20201103 张三 did not take any exams
202011 has no grades yet
输入样例6:
单门课程,多个学生。例如:
查看代码 java 选修 考察
20201103 李四 java 60
20201104 王五 java 60
20201101 张三 java 40
end
输出样例6:
在这里给出相应的输出。例如:
查看代码 20201101 张三 40
20201103 李四 60
20201104 王五 60
java 53 53
202011 53
输入样例7:
单门课程,单个学生,课程类型与考核类型不匹配。例如:
查看代码 形式与政治 必修 考试
数据库 选修 考试
java 选修 考察
数据结构 选修 考察
20201103 李四 数据结构 70
20201103 李四 形式与政治 80 90
20201103 李四 java 60
20201103 李四 数据库 70 78
end
输出样例7:
在这里给出相应的输出。例如:
查看代码 20201103 李四 73
java 60 60
数据结构 70 70
数据库 70 78 75
形式与政治 80 90 87
202011 73
输入样例8:
单门课程,单个学生,成绩越界。例如:
查看代码 数据结构 选修 考察
20201103 李四 数据结构 101
end
输出样例8:
在这里给出相应的输出。例如:
查看代码 wrong format
数据结构 has no grades yet
输入样例9:
多门课程,多个学生,多个成绩。例如:
查看代码 形式与政治 必修 考试
数据库 选修 考试
java 选修 考察
数据结构 选修 考察
20201205 李四 数据结构 70
20201103 李四 形式与政治 80 90
20201102 王五 java 60
20201211 张三 数据库 70 78
end
输出样例9:
在这里给出相应的输出。例如:
查看代码 20201102 王五 60
20201103 李四 87
20201205 李四 70
20201211 张三 75
java 60 60
数据结构 70 70
数据库 70 78 75
形式与政治 80 90 87
202011 73
202012 72
设计与分析:
这道题是第一次迭代,从零开始会比较麻烦,花的时间也比较多。和上题的判断输入信息的处理差不多,都是一行行读,对每行的信息进行判断处理,使用正则,然后就用空格分隔字符串,再进行判断,对应的有很多逻辑问题和错误类型,就比如要在什么时候存入信息,存入信息的条件是什么,这题过不去是有很多情况没考虑到,只考虑了简单的几种情况,一复杂就对不了。选课系统的课程起到一个类似菜单的作用,在存入成绩信息的时候是创建一个新的课程对象,赋以选课系统里课程的信息以及成绩信息加到学生类的课程对象数组里,学生再加到班级类的学生对象数组里。最后根据选课系统里的课程输出各种数据。对成绩的计算是先在各个成绩类中算出来再在学生类中汇总成学生成绩,再汇总到班级中。对于各门课程的成绩是先在选课系统的课程的循环中遍历学生,再遍历学生的课程,如果有对应的课程就加入,直到最后没有就输出无成绩。
类图:

类图中该聚合的没有聚合,说明都在主函数里有联系,类之间却没有。
SourceMonitor的生成报表内容:

采坑心得:
重复的课程名那个测试点自己试出来了是不对的,但是当时没时间再改了。写这种类似系统的基本都是逻辑错和考虑不周全,还是要自己多设测试点。重复的课程名有错,输出了两个一样的课程信息,就是因为在判断是重复的课程名之后还将课程信息存进去了。

改进建议:
再对课程信息的处理那里检查一下,对课程的存入多设置条件。
源码如下:
查看代码 import java.util.Collections;
import java.util.Scanner;
import java.util.ArrayList;
import java.text.Collator;
import java.util.Comparator;
import java.util.Locale;
public class Main {
public static void main(String[] args) {
Scanner input = new Scanner(System.in);
Choosecourses choosecourses = new Choosecourses();
String courseMessage = input.nextLine();
while (!courseMessage.startsWith("end")){
if(checkCourseMessage(courseMessage)){
String[] message = courseMessage.split(" ");
Course course = new Course();
course.name = message[0];
course.character = message[1];
if(!choosecourses.findCourse(course.name)){
if(message.length == 3){
if(message[1].equals("必修")&&message[2].equals("考察")){
System.out.println(message[0]+" : course type & access mode mismatch");
}else{
course.way = message[2];
choosecourses.getCourses().add(course);
}
}else if(message[1].equals("选修")){
System.out.println(message[0]+" : course type & access mode mismatch");
}else{
course.way = "考试";
choosecourses.getCourses().add(course);
}
}
}else{
System.out.println("wrong format");
}
courseMessage = input.nextLine();
if(checkScoreMessage(courseMessage)){
break;
}
}
String scoreMessage = courseMessage;
ArrayList<Student> students = new ArrayList<>();
while(!scoreMessage.startsWith("end")){
if(checkScoreMessage(scoreMessage)){
String[] message = scoreMessage.split(" ");
String classNumber = message[0].substring(0,6);
int flag = 0;
int flag1 = 0;
Course course1 = new Course();
for (Course course :choosecourses.getCourses()) {
if(course.name.equals(message[2])){
flag = 1;
if(message.length == 5&&course.way.equals("考试")){
course1.examScore.usualGrade = (Integer.parseInt(message[3]));
course1.examScore.score = Integer.parseInt(message[4]);
course1.way = course.way;
course1.name = course.name;
flag1 = 2;
}else if(message.length == 4&&course.way.equals("考察")){
course1.inspectScore.score = Integer.parseInt(message[3]);
course1.way = course.way;
course1.name = course.name;
flag1 = 2;
}else{
System.out.println(message[0]+" "+message[1]+" "+": access mode mismatch");
}
break;
}
}
if(choosecourses.findStudent(message[0]) == null){
Student student = new Student();
student.studentNumber = message[0];
student.name = message[1];
if(flag1 == 2){
student.courses.add(course1);
}
choosecourses.getStudent().add(student);
if(choosecourses.findClass(classNumber) == null){
Classes classes = new Classes(classNumber);
classes.students.add(student);
choosecourses.getClasses().add(classes);
}else{
choosecourses.findClass(classNumber).students.add(student);
}
}else if(choosecourses.findStudent(message[0]).findCourse(message[2]) == null){
choosecourses.findStudent(message[0]).courses.add(course1);
}
if(flag == 0){
System.out.println(message[2]+" "+"does not exist");
}
}else {
System.out.println("wrong format");
}
scoreMessage = input.nextLine();
if(!checkScoreMessage(scoreMessage)){
break;
}
}
for(int i = 0;i < choosecourses.getStudent().size();i ++){
for(int j = 0;j < choosecourses.getStudent().size() - i - 1;j ++){
if(choosecourses.getStudent().get(j).studentNumber.compareTo(choosecourses.getStudent().get(j + 1).studentNumber)>0){
Student student = choosecourses.getStudent().get(j + 1);
choosecourses.getStudent().set(j + 1,choosecourses.getStudent().get(j));
choosecourses.getStudent().set(j,student);
}
}
}
for (Student student:choosecourses.getStudent()) {
if(student.courses.size() == 0){
System.out.println(student.studentNumber+" "+student.name+" did not take any exams");
}else{
System.out.println(student.studentNumber+" "+student.name+" "+student.getTotallAverage());
}
}
Collections.sort(choosecourses.getCourses());
for (Course course:choosecourses.getCourses()) {
if(choosecourses.findCourse(course.name)){
if(course.way.equals("考试")){
System.out.println(course.name+" "+choosecourses.getCourseAverageUsualGrade(course)+" "+
choosecourses.getCourseAverageFinalGrade(course)+" "+choosecourses.getCourseAverageTotallGrade(course));
}else{
System.out.println(course.name+" "+choosecourses.getCourseAverageFinalGrade(course)+" "+
choosecourses.getCourseAverageTotallGrade(course));
}
}else{
System.out.println(course.name+" has no grades yet");
}
}
for(int i = 0;i < choosecourses.getClasses().size();i ++){
for(int j = 0;j < choosecourses.getClasses().size() - i - 1;j ++){
if(choosecourses.getClasses().get(j).number.compareTo(choosecourses.getClasses().get(j + 1).number)>0){
Classes classes = choosecourses.getClasses().get(j + 1);
choosecourses.getClasses().set(j + 1,choosecourses.getClasses().get(j));
choosecourses.getClasses().set(j,classes);
}
}
}
for (Classes classes:choosecourses.getClasses()) {
if(classes.isCourseEmpty()){
System.out.println(classes.number+" has no grades yet");
}else {
System.out.println(classes.number+" "+classes.getTotallAverage());
}
}
}
public static boolean checkCourseMessage(String courseMessage){
if(courseMessage.matches(".{1,11}\\s(必修|选修)\\s(考试|考察)")||courseMessage.matches(".{1,11}\\s(必修)")){
return true;
}
return false;
}
public static boolean checkScoreMessage(String scoreMessage){
if(scoreMessage.matches("[\\d]{8}\\s.{1,11}\\s.{1,11}\\s([0-9]|[1-9][0-9]|100)")
||scoreMessage.matches("[\\d]{8}\\s.{1,11}\\s.{1,11}\\s([0-9]|[1-9][0-9]|100)\\s([0-9]|[1-9][0-9]|100)"))
return true;
return false;
}
}
abstract class AbstractScore {
int score;
public AbstractScore(int score) {
this.score = score;
}
public AbstractScore() {
}
public int getScore() {
return score;
}
public void setScore(int score) {
this.score = score;
}
public int getTotall(){
return score;
}
}
class Choosecourses {
private ArrayList<Course> courses = new ArrayList<>();
private ArrayList<Student> students = new ArrayList<>();
private ArrayList<AbstractScore> scores = new ArrayList<>();
private ArrayList<Classes> classes = new ArrayList<>();
public ArrayList<Classes> getClasses() {
return classes;
}
public void setClasses(ArrayList<Classes> classes) {
this.classes = classes;
}
public ArrayList<Course> getCourses() {
return courses;
}
public void setCourses(ArrayList<Course> courses) {
this.courses = courses;
}
public ArrayList<Student> getStudent() {
return students;
}
public void setStudent(ArrayList<Student> student) {
this.students = student;
}
public Choosecourses() {
}
public int getCourseAverageTotallGrade(Course course){//单门课程平均总成绩
int sum = 0;
int i = 0;
for (Student student:students) {
for (Course course1: student.courses) {
if(course1.name.equals(course.name)){
sum += course1.getGrade();
i ++;
break;
}
}
}
if(i != 0)
return sum/i;
return 0;
}
public int getCourseAverageUsualGrade(Course course){//单门课程平均平时成绩
int sum = 0;
int i = 0;
for (Student student:students) {
for (Course course1: student.courses) {
if(course1.name.equals(course.name)){
sum += course1.examScore.usualGrade;
i ++;
break;
}
}
}
if(i != 0){
return sum/i;
}
return 0;
}
public int getCourseAverageFinalGrade(Course course){//单门课程平均期末成绩
int sum = 0;
int i = 0;
for (Student student:students) {
for (Course course1: student.courses) {
if(course1.name.equals(course.name)){
if(course1.way.equals("考试")){
sum += course1.examScore.score;
}else{
sum += course1.inspectScore.score;
}
i ++;
break;
}
}
}
if(i != 0)
return sum/i;
return 0;
}
public boolean findCourse(String course){
for (Student student: students) {
for (Course course1: student.getCourses()) {
if(course1.name.equals(course)){
return true;
}
}
}
return false;
}
public Student findStudent(String str){
for (Student stu: students) {
if(stu.getStudentNumber().equals(str)){
return stu;
}
}
return null;
}
public Classes findClass(String str){
for (Classes class1: classes) {
if(class1.number.equals(str)){
return class1;
}
}
return null;
}
}
class Classes {
ArrayList<Student> students = new ArrayList<>();
String number;
public Classes() {
}
public Classes(String number) {
this.number = number;
}
public int getTotallAverage(){//班级总成绩平均分
int sum = 0;
for (Student student: students) {
sum += student.getTotallAverage();
}
return sum/students.size();
}
public boolean isCourseEmpty(){
for (Student stu: students) {
if(!stu.courses.isEmpty()){
return false;
}
}
return true;
}
}
class Course implements Comparable<Course>{
String name;
String character;
String way;
ExamScore examScore = new ExamScore();
InspectScore inspectScore = new InspectScore();
public Course(String name, String character, String way, ExamScore examScore, InspectScore inspectScore) {
this.name = name;
this.character = character;
this.way = way;
this.examScore = examScore;
this.inspectScore = inspectScore;
}
public Course() {
}
public int getGrade(){
if(way.equals("考察")){
return inspectScore.getTotall();
}else {
return examScore.getTotall();
}
}
@Override
public int compareTo(Course course) {
Comparator<Object> comparator = Collator.getInstance(Locale.CHINA);
return ((Collator) comparator).compare(this.name,course.name); //升序
}
}
class ExamScore extends AbstractScore{
int usualGrade;
public ExamScore(int score) {
super(score);
}
public ExamScore() {
}
@Override
public int getTotall(){
return (int)(usualGrade * 0.3 + score * 0.7);
}
}
class InspectScore extends AbstractScore{
public InspectScore(int score) {
super(score);
}
public InspectScore() {
}
}
class Student {
String studentNumber;
String name;
ArrayList<Course> courses = new ArrayList<>();
public String getStudentNumber() {
return studentNumber;
}
public void setStudentNumber(String studentNumber) {
this.studentNumber = studentNumber;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public ArrayList<Course> getCourses() {
return courses;
}
public void setCourses(ArrayList<Course> courses) {
this.courses = courses;
}
public Student() {
}
public Student(String studentNumber, String name, ArrayList<Course> courses) {
this.studentNumber = studentNumber;
this.name = name;
this.courses = courses;
}
public int getTotallAverage(){//所有课程总成绩平均分
int sum = 0;
for (Course course :courses) {
sum += course.getGrade();
}
if(courses.size() != 0)
return sum/courses.size();
return 0;
}
public Course findCourse(String name){
for (Course course :courses) {
if(course.name.equals(name)){
return course;
}
}
return null;
}
}
题目集09:
编写程序统计一个输入的Java源码中关键字(区分大小写)出现的次数。说明如下:
- Java中共有53个关键字(自行百度)
- 从键盘输入一段源码,统计这段源码中出现的关键字的数量
- 注释中出现的关键字不用统计
- 字符串中出现的关键字不用统计
- 统计出的关键字及数量按照关键字升序进行排序输出
- 未输入源码则认为输入非法
输入格式:
输入Java源码字符串,可以一行或多行,以exit行作为结束标志
输出格式:
- 当未输入源码时,程序输出
Wrong Format - 当没有统计数据时,输出为空
- 当有统计数据时,关键字按照升序排列,每行输出一个关键字及数量,格式为
数量\t关键字
输入样例:
在这里给出一组输入。例如:
查看代码 //Test public method
public HashMap(int initialCapacity) {
this(initialCapacity, DEFAULT_LOAD_FACTOR);
}
public HashMap(int initialCapacity, float loadFactor) {
if (initialCapacity < 0)
throw new IllegalArgumentException("Illegal initial capacity: " +
initialCapacity);
if (initialCapacity > MAXIMUM_CAPACITY)
initialCapacity = MAXIMUM_CAPACITY;
if (loadFactor <= 0 || Float.isNaN(loadFactor))
throw new IllegalArgumentException("Illegal load factor: " +
loadFactor);
this.loadFactor = loadFactor;
this.threshold = tableSizeFor(initialCapacity);
}
exit
输出样例:
在这里给出相应的输出。例如:
查看代码 1 float
3 if
2 int
2 new
2 public
3 this
2 throw
设计与分析:
这题之前是用的一行一行判断,有点繁琐,看了下网上的方法,是先对输进来的代码简化,先把注释里和字符串里的东西都先替换成括号,再用正则判断就容易多了。


采坑心得:
之前的方法比较繁琐,判断比较复杂,但可以先对信息简化之后再判断,有时候这样会轻松许多。
改进建议:
如果还需要输出原代码的话可以在预处理之前先把每句直接存在一个容器里,最后输出来。
源码如下:
查看代码 import java.util.HashSet;
import java.util.Map;
import java.util.Scanner;
import java.util.TreeMap;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
public class Main {
public static void main(String[] args) {
String[] strings = {"abstract","assert","boolean","break","byte","case","catch","char","class","const","continue","default","do",
"double","else","enum","extends","final","finally","float","for","goto","if","implements","import","instanceof","int",
"interface","long","native","new","package","private","protected","public","return", "strictfp","short","static","super",
"switch","synchronized","this","throw","throws","transient","try","void","volatile","while","true","false","null"};
Scanner input = new Scanner(System.in);
TreeMap<String,Integer> treeMap = new TreeMap<>();
for (String s1: strings) {
treeMap.put(s1,0);
}
int flag = 0;
String a = "";
String s = input.nextLine();
while (!s.trim().startsWith("exit")){
if(!s.equals("")){
flag = 1;
if(s.contains("/*")){
a = a + s.replaceAll("/\\*.*"," ");
s = input.nextLine();
while (!s.contains("*/")){
s = input.nextLine();
}
a = a + s.replaceAll(".*\\*/"," ");
}else {
a = a + (s.replaceAll("//.*", " ").replaceAll("\".*\"", " "));
s = input.nextLine();
}
}else {
s = input.nextLine();
}
}
a = a.replaceAll("=",1+"");
for (String s1: strings) {
Pattern p = Pattern.compile("\\b"+s1+"\\b");
Matcher m = p.matcher(a);
int i=0;
while (m.find()){
i++;
}
treeMap.put(s1,i);
}
if(flag == 0){
System.out.println("Wrong Format");
}else{
for (Map.Entry<String, Integer> entry : treeMap.entrySet()) {
if(entry.getValue() != 0){
System.out.println(entry.getValue() + "\t" + entry.getKey());
}
}
}
}
}
题目集10:
课程成绩统计程序-2在第一次的基础上增加了实验课,以下加粗字体显示为本次新增的内容。
某高校课程从性质上分为:必修课、选修课、实验课,从考核方式上分为:考试、考察、实验。
考试的总成绩由平时成绩、期末成绩分别乘以权重值得出,比如平时成绩权重0.3,期末成绩权重0.7,总成绩=平时成绩*0.3+期末成绩*0.7。
考察的总成绩直接等于期末成绩
实验的总成绩等于课程每次实验成绩的平均分
必修课的考核方式必须为考试,选修课可以选择考试、考察任一考核方式。实验课的成绩必须为实验。
1、输入:
包括课程、课程成绩两类信息。
课程信息包括:课程名称、课程性质、考核方式(可选,如果性质是必修课,考核方式可以没有)三个数据项。
课程信息格式:课程名称+英文空格+课程性质+英文空格+考核方式
课程性质输入项:必修、选修、实验
考核方式输入选项:考试、考察、实验
考试/考查课程成绩信息包括:学号、姓名、课程名称、平时成绩(可选)、期末成绩
考试/考查课程信息格式:学号+英文空格+姓名+英文空格+课程名称+英文空格+平时成绩+英文空格+期末成绩
实验课程成绩信息包括:学号、姓名、课程名称、实验次数、每次成绩
实验次数至少4次,不超过9次
实验课程信息格式:学号+英文空格+姓名+英文空格+课程名称+英文空格+实验次数+英文空格+第一次实验成绩+...+英文空格+最后一次实验成绩
以上信息的相关约束:
1)平时成绩和期末成绩的权重默认为0.3、0.7
2)成绩是整数,不包含小数部分,成绩的取值范围是【0,100】
3)学号由8位数字组成
4)姓名不超过10个字符
5)课程名称不超过10个字符
6)不特别输入班级信息,班级号是学号的前6位。
2、输出:
输出包含三个部分,包括学生所有课程总成绩的平均分、单门课程成绩平均分、单门课程总成绩平均分、班级所有课程总成绩平均分。
为避免误差,平均分的计算方法为累加所有符合条件的单个成绩,最后除以总数。
1)学生课程总成绩平均分按学号由低到高排序输出
格式:学号+英文空格+姓名+英文空格+总成绩平均分
如果某个学生没有任何成绩信息,输出:学号+英文空格+姓名+英文空格+"did not take any exams"
2)单门课程成绩平均分分为三个分值:平时成绩平均分(可选)、期末考试平均分、总成绩平均分,按课程名称的字符顺序输出
考试/考察课程成绩格式:课程名称+英文空格+平时成绩平均分+英文空格+期末考试平均分+英文空格+总成绩平均分
实验课成绩格式:课程名称+英文空格+总成绩平均分
如果某门课程没有任何成绩信息,输出:课程名称+英文空格+"has no grades yet"
3)班级所有课程总成绩平均分按班级由低到高排序输出
格式:班级号+英文空格+总成绩平均分
如果某个班级没有任何成绩信息,输出:班级名称+英文空格+ "has no grades yet"
异常情况:
1)如果解析某个成绩信息时,课程名称不在已输入的课程列表中,输出:学号+英文空格+姓名+英文空格+":"+课程名称+英文空格+"does not exist"
2)如果解析某个成绩信息时,输入的成绩数量和课程的考核方式不匹配,输出:学号+英文空格+姓名+英文空格+": access mode mismatch"
以上两种情况如果同时出现,按第一种情况输出结果。
3)如果解析某个课程信息时,输入的课程性质和课程的考核方式不匹配,输出:课程名称+" : course type & access mode mismatch"
4)格式错误以及其他信息异常如成绩超出范围等,均按格式错误处理,输出"wrong format"
5)若出现重复的课程/成绩信息,只保留第一个课程信息,忽略后面输入的。
信息约束:
1)成绩平均分只取整数部分,小数部分丢弃
参考类图(与第一次相同,其余内容自行补充):

输入样例1:
在这里给出一组输入。例如:
查看代码 java 实验 实验
20201103 张三 java 4 70 80 90
end
输出样例1:
在这里给出相应的输出。例如:
查看代码 20201103 张三 : access mode mismatch
20201103 张三 did not take any exams
java has no grades yet
202011 has no grades yet
输入样例2:
在这里给出一组输入。例如:
查看代码 java 实验 实验
20201103 张三 java 3 70 80 90
end
输出样例2:
在这里给出相应的输出。例如:
查看代码 wrong format
java has no grades yet
输入样例3:
在这里给出一组输入。例如:
查看代码 java 必修 实验
20201103 张三 java 3 70 80 90 100
end
输出样例3:
在这里给出相应的输出。例如:
查看代码 java : course type & access mode mismatch
wrong format
输入样例4:
在这里给出一组输入。例如:
查看代码 java 必修 实验
20201103 张三 java 4 70 80 90 105
end
输出样例4:
在这里给出相应的输出。例如:
查看代码 java : course type & access mode mismatch
wrong format
输入样例5:
在这里给出一组输入。例如:
查看代码 java 选修 考察
C语言 选修 考察
java实验 实验 实验
编译原理 必修 考试
20201101 王五 C语言 76
20201216 李四 C语言 78
20201307 张少军 编译原理 82 84
20201103 张三 java实验 4 70 80 90 100
20201118 郑觉先 java 80
20201328 刘和宇 java 77
20201220 朱重九 java实验 4 60 60 80 80
20201132 王萍 C语言 40
20201302 李梦涵 C语言 68
20201325 崔瑾 编译原理 80 84
20201213 黄红 java 82
20201209 赵仙芝 java 76
end
输出样例5:
在这里给出相应的输出。例如:
查看代码 20201101 王五 76
20201103 张三 85
20201118 郑觉先 80
20201132 王萍 40
20201209 赵仙芝 76
20201213 黄红 82
20201216 李四 78
20201220 朱重九 70
20201302 李梦涵 68
20201307 张少军 83
20201325 崔瑾 82
20201328 刘和宇 77
C语言 65 65
java 78 78
java实验 77
编译原理 81 84 82
202011 70
202012 76
202013 77
设计与分析:
这题是之前的迭代,加了实验课,所以在课程信息和成绩信息的处理和输出要改一下,加了实验课就代表着会有许多和必修课选修课混杂在一起错误的问题,情况比较多,针对实验课的处理也加了一些判定条件
这里对很多种组合的类型放一起判断,后面就不用再考虑这一方面。
类图:

类图差别与上次不大,多了实验课相关的东西。
SourceMonitor的生成报表内容:

采坑心得:
后面自己设置测试点发现在一门课程多个学生时学生成绩都变成最后一个人的成绩了。

原因就是这里的course1和course的赋值问题,这里不能直接将二者相等来给他们对应的属性赋相同的值,应该要分别对属性进行赋值才行。以后对两个对象的赋值也要注意这点。
改进建议:
多去实现单一职责原则,还有就是主函数里的东西太多了,可以试着简化一下。
源码如下:
查看代码 import java.util.Collections;
import java.util.Scanner;
import java.util.ArrayList;
import java.text.Collator;
import java.util.Comparator;
import java.util.Locale;
public class Main {
public static void main(String[] args) {
Scanner input = new Scanner(System.in);
Choosecourses choosecourses = new Choosecourses();
String courseMessage = input.nextLine();
while (!courseMessage.startsWith("end")){
if(checkCourseMessage(courseMessage)){
String[] message = courseMessage.split(" ");
Course course = new Course();
course.name = message[0];
course.character = message[1];
if(!choosecourses.findCourse(course.name)){
if(message.length == 3){
if((message[1].equals("必修")&& (message[2].equals("考察")||message[2].equals("实验")) ) ||
(message[1].equals("实验")&& (message[2].equals("考试")||message[2].equals("考察")) ) ||
(message[1]).equals("选修")&&message[2].equals("实验") ){
System.out.println(message[0]+" : course type & access mode mismatch");
}else{
course.way = message[2];
choosecourses.getCourses().add(course);
}
}else if(message[1].equals("选修")){
System.out.println(message[0]+" : course type & access mode mismatch");
}else if(message[1].equals("必修")){
course.way = "考试";
choosecourses.getCourses().add(course);
}else {
course.way = "实验";
choosecourses.getCourses().add(course);
}
}
}else{
System.out.println("wrong format");
}
courseMessage = input.nextLine();
if(checkScoreMessage(courseMessage)){
break;
}
}
String scoreMessage = courseMessage;
while(!scoreMessage.startsWith("end")){
if(checkScoreMessage(scoreMessage)){
String[] message = scoreMessage.split(" ");
String classNumber = message[0].substring(0,6);
int flag = 0;
int flag1 = 0;
int flag2 = 0;
Course course1 = new Course();
for (Course course :choosecourses.getCourses()) {
if(course.name.equals(message[2])){
flag = 1;
if(message.length == 5&&course.way.equals("考试")){
course1.examScore.usualGrade = (Integer.parseInt(message[3]));
course1.examScore.score = Integer.parseInt(message[4]);
course1.way = course.way;
course1.name = course.name;
flag1 = 2;
}else if(message.length == 4&&course.way.equals("考察")){
course1.inspectScore.score = Integer.parseInt(message[3]);
course1.way = course.way;
course1.name = course.name;
flag1 = 2;
}else if(course.way.equals("实验")&&(Integer.parseInt(message[3]) < 4||Integer.parseInt(message[3]) > 9)){
System.out.println("wrong format");
flag2 = 1;
}else if(message.length == 4 + Integer.parseInt(message[3])&&course.way.equals("实验")){
course1.way = course.way;
course1.name = course.name;
for(int i = 4;i < message.length;i++){
course1.experimentScore.scores.add(message[i]);
}
flag1 = 2;
}else {
System.out.println(message[0]+" "+message[1]+" "+": access mode mismatch");
}
break;
}
}
if(flag2 == 0){
if(choosecourses.findStudent(message[0]) == null){
Student student = new Student();
student.studentNumber = message[0];
student.name = message[1];
if(flag1 == 2){
student.courses.add(course1);
}
choosecourses.getStudent().add(student);
if(choosecourses.findClass(classNumber) == null){
Classes classes = new Classes(classNumber);
classes.students.add(student);
choosecourses.getClasses().add(classes);
}else{
choosecourses.findClass(classNumber).students.add(student);
}
}else if(choosecourses.findStudent(message[0]).findCourse(message[2]) == null&&flag1 == 2){
choosecourses.findStudent(message[0]).courses.add(course1);
}
}
if(flag == 0){
System.out.println(message[2]+" "+"does not exist");
}
}else {
System.out.println("wrong format");
}
scoreMessage = input.nextLine();
}
for(int i = 0;i < choosecourses.getStudent().size();i ++){
for(int j = 0;j < choosecourses.getStudent().size() - i - 1;j ++){
if(choosecourses.getStudent().get(j).studentNumber.compareTo(choosecourses.getStudent().get(j + 1).studentNumber)>0){
Student student = choosecourses.getStudent().get(j + 1);
choosecourses.getStudent().set(j + 1,choosecourses.getStudent().get(j));
choosecourses.getStudent().set(j,student);
}
}
}
for (Student student:choosecourses.getStudent()) {
if(student.courses.size() == 0){
System.out.println(student.studentNumber+" "+student.name+" did not take any exams");
}else{
System.out.println(student.studentNumber+" "+student.name+" "+student.getTotallAverage());
}
}
Collections.sort(choosecourses.getCourses());
for (Course course:choosecourses.getCourses()) {
if(choosecourses.findCourse1(course.name)){
if(course.way.equals("考试")){
System.out.println(course.name+" "+choosecourses.getCourseAverageUsualGrade(course)+" "+
choosecourses.getCourseAverageFinalGrade(course)+" "+choosecourses.getCourseAverageTotallGrade(course));
}else if(course.way.equals("考察")){
System.out.println(course.name+" "+choosecourses.getCourseAverageFinalGrade(course)+" "+
choosecourses.getCourseAverageTotallGrade(course));
}else {
System.out.println(course.name+" "+choosecourses.getCourseAverageTotallGrade(course));
}
}else{
System.out.println(course.name+" has no grades yet");
}
}
for(int i = 0;i < choosecourses.getClasses().size();i ++){
for(int j = 0;j < choosecourses.getClasses().size() - i - 1;j ++){
if(choosecourses.getClasses().get(j).number.compareTo(choosecourses.getClasses().get(j + 1).number)>0){
Classes classes = choosecourses.getClasses().get(j + 1);
choosecourses.getClasses().set(j + 1,choosecourses.getClasses().get(j));
choosecourses.getClasses().set(j,classes);
}
}
}
for (Classes classes:choosecourses.getClasses()) {
if(classes.isCourseEmpty()){
System.out.println(classes.number+" has no grades yet");
}else {
System.out.println(classes.number+" "+classes.getTotallAverage());
}
}
}
public static boolean checkCourseMessage(String courseMessage){
if(courseMessage.matches(".{1,11}\\s(必修|选修|实验)\\s(考试|考察|实验)")||courseMessage.matches(".{1,11}\\s(必修|实验)")){
return true;
}
return false;
}
public static boolean checkScoreMessage(String scoreMessage){
if(scoreMessage.matches("[\\d]{8}\\s.{1,11}\\s.{1,11}\\s[456789](\\s([0-9]|[1-9][0-9]|100))+")
||scoreMessage.matches("[\\d]{8}\\s.{1,11}\\s.{1,11}\\s([0-9]|[1-9][0-9]|100)")
||scoreMessage.matches("[\\d]{8}\\s.{1,11}\\s.{1,11}\\s([0-9]|[1-9][0-9]|100)\\s([0-9]|[1-9][0-9]|100)"))
return true;
return false;
}
}
abstract class AbstractScore {
int score;
public AbstractScore(int score) {
this.score = score;
}
public AbstractScore() {
}
public int getScore() {
return score;
}
public void setScore(int score) {
this.score = score;
}
public int getTotall(){
return score;
}
}
class Choosecourses {
private ArrayList<Course> courses = new ArrayList<>();
private ArrayList<Student> students = new ArrayList<>();
private ArrayList<AbstractScore> scores = new ArrayList<>();
private ArrayList<Classes> classes = new ArrayList<>();
public ArrayList<Classes> getClasses() {
return classes;
}
public void setClasses(ArrayList<Classes> classes) {
this.classes = classes;
}
public ArrayList<Course> getCourses() {
return courses;
}
public void setCourses(ArrayList<Course> courses) {
this.courses = courses;
}
public ArrayList<Student> getStudent() {
return students;
}
public void setStudent(ArrayList<Student> student) {
this.students = student;
}
public Choosecourses() {
}
public int getCourseAverageTotallGrade(Course course){//单门课程平均总成绩
int sum = 0;
int i = 0;
for (Student student:students) {
for (Course course1: student.courses) {
if(course1.name.equals(course.name)){
sum += course1.getGrade();
i ++;
break;
}
}
}
if(i != 0)
return sum/i;
return 0;
}
public int getCourseAverageUsualGrade(Course course){//单门课程平均平时成绩
int sum = 0;
int i = 0;
for (Student student:students) {
for (Course course1: student.courses) {
if(course1.name.equals(course.name)){
sum += course1.examScore.usualGrade;
i ++;
break;
}
}
}
if(i != 0){
return sum/i;
}
return 0;
}
public int getCourseAverageFinalGrade(Course course){//单门课程平均期末成绩
int sum = 0;
int i = 0;
for (Student student:students) {
for (Course course1: student.courses) {
if(course1.name.equals(course.name)){
if(course1.way.equals("考试")){
sum += course1.examScore.score;
}else{
sum += course1.inspectScore.score;
}
i ++;
break;
}
}
}
if(i != 0)
return sum/i;
return 0;
}
public boolean findCourse(String str){
for (Course course: courses) {
if(course.name.equals(str)){
return true;
}
}
return false;
}
public boolean findCourse1(String course){
for (Student student: students) {
for (Course course1: student.courses) {
if(course1.name.equals(course)){
return true;
}
}
}
return false;
}
public Student findStudent(String str){
for (Student stu: students) {
if(stu.studentNumber.equals(str)){
return stu;
}
}
return null;
}
public Classes findClass(String str){
for (Classes class1: classes) {
if(class1.number.equals(str)){
return class1;
}
}
return null;
}
}
class Classes {
ArrayList<Student> students = new ArrayList<>();
String number;
public Classes() {
}
public Classes(String number) {
this.number = number;
}
public int getTotallAverage(){//班级总成绩平均分
int sum = 0;
for (Student student: students) {
sum += student.getTotallAverage();
}
return sum/students.size();
}
public boolean isCourseEmpty(){
for (Student stu: students) {
if(!stu.courses.isEmpty()){
return false;
}
}
return true;
}
}
class Course implements Comparable<Course>{
String name;
String character;
String way;
ExamScore examScore = new ExamScore();
InspectScore inspectScore = new InspectScore();
ExperimentScore experimentScore = new ExperimentScore();
public Course(String name, String character, String way, ExamScore examScore, InspectScore inspectScore) {
this.name = name;
this.character = character;
this.way = way;
this.examScore = examScore;
this.inspectScore = inspectScore;
}
public Course() {
}
public int getGrade(){
if(way.equals("考察")){
return inspectScore.getTotall();
}else if(way.equals("实验")){
return experimentScore.getTotall();
}else {
return examScore.getTotall();
}
}
@Override
public int compareTo(Course course) {
Comparator<Object> comparator = Collator.getInstance(Locale.CHINA);
return ((Collator) comparator).compare(this.name,course.name); //升序
}
}
class ExamScore extends AbstractScore{
int usualGrade;
public ExamScore(int score) {
super(score);
}
public ExamScore() {
}
@Override
public int getTotall(){
return (int)(usualGrade * 0.3 + score * 0.7);
}
}
class ExperimentScore extends AbstractScore{
ArrayList<String> scores = new ArrayList<>();
@Override
public int getTotall(){
int sum = 0;
for (String s: scores) {
sum += Integer.parseInt(s);
}
score = sum/scores.size();
return score;
}
public ExperimentScore(int score) {
super(score);
}
public ExperimentScore() {
}
}
class InspectScore extends AbstractScore{
public InspectScore(int score) {
super(score);
}
public InspectScore() {
}
@Override
public int getTotall(){
return score;
}
}
class Student {
String studentNumber;
String name;
ArrayList<Course> courses = new ArrayList<>();
public Student() {
}
public Student(String studentNumber, String name, ArrayList<Course> courses) {
this.studentNumber = studentNumber;
this.name = name;
this.courses = courses;
}
public int getTotallAverage(){//所有课程总成绩平均分
int sum = 0;
for (Course course :courses) {
sum += course.getGrade();
}
if(courses.size() != 0)
return sum/courses.size();
return 0;
}
public Course findCourse(String name){
for (Course course :courses) {
if(course.name != null&&course.name.equals(name)){
return course;
}
}
return null;
}
}
题目集11:
课程成绩统计程序-3在第二次的基础上修改了计算总成绩的方式,
要求:修改类结构,将成绩类的继承关系改为组合关系,成绩信息由课程成绩类和分项成绩类组成,课程成绩类组合分项成绩类,分项成绩类由成绩分值和权重两个属性构成。
完成课程成绩统计程序-2、3两次程序后,比较继承和组合关系的区别。思考一下哪一种关系运用上更灵活,更能够适应变更。
题目最后的参考类图未做修改,大家根据要求自行调整,以下内容加粗字体显示的内容为本次新增的内容。
某高校课程从性质上分为:必修课、选修课、实验课,从考核方式上分为:考试、考察、实验。
考试的总成绩由平时成绩、期末成绩分别乘以权重值得出,比如平时成绩权重0.3,期末成绩权重0.7,总成绩=平时成绩*0.3+期末成绩*0.7。
考察的总成绩直接等于期末成绩
实验的总成绩等于课程每次实验成绩乘以权重后累加而得。
课程权重值在录入课程信息时输入。(注意:所有分项成绩的权重之和应当等于1)
必修课的考核方式必须为考试,选修课可以选择考试、考察任一考核方式。实验课的成绩必须为实验。
1、输入:
包括课程、课程成绩两类信息。
课程信息包括:课程名称、课程性质、考核方式、分项成绩数量、每个分项成绩的权重。
考试课信息格式:课程名称+英文空格+课程性质+英文空格+考核方式+英文空格+平时成绩的权重+英文空格+期末成绩的权重
考察课信息格式:课程名称+英文空格+课程性质+英文空格+考核方式
实验课程信息格式:课程名称+英文空格+课程性质+英文空格+考核方式+英文空格+分项成绩数量n+英文空格+分项成绩1的权重+英文空格+。。。+英文空格+分项成绩n的权重
实验次数至少4次,不超过9次
课程性质输入项:必修、选修、实验
考核方式输入选项:考试、考察、实验
考试/考查课程成绩信息包括:学号、姓名、课程名称、平时成绩(可选)、期末成绩
考试/考查课程成绩信息格式:学号+英文空格+姓名+英文空格+课程名称+英文空格+平时成绩+英文空格+期末成绩
实验课程成绩信息包括:学号、姓名、课程名称、每次成绩{在系列-2的基础上去掉了(实验次数),实验次数要和实验课程信息中输入的分项成绩数量保持一致}
实验课程信息格式:学号+英文空格+姓名+英文空格+课程名称+英文空格+第一次实验成绩+...+英文空格+最后一次实验成绩
以上信息的相关约束:
1)成绩是整数,不包含小数部分,成绩的取值范围是【0,100】
2)学号由8位数字组成
3)姓名不超过10个字符
4)课程名称不超过10个字符
5)不特别输入班级信息,班级号是学号的前6位。
2、输出:
输出包含三个部分,包括学生所有课程总成绩的平均分、单门课程总成绩平均分、班级所有课程总成绩平均分。
为避免四舍五入误差,
计算单个成绩时,分项成绩乘以权重后要保留小数位,计算总成绩时,累加所有分项成绩的权重分以后,再去掉小数位。
学生总成绩/整个班/课程平均分的计算方法为累加所有符合条件的单个成绩,最后除以总数。
1)学生课程总成绩平均分按学号由低到高排序输出
格式:学号+英文空格+姓名+英文空格+总成绩平均分
如果某个学生没有任何成绩信息,输出:学号+英文空格+姓名+英文空格+"did not take any exams"
2)单门课程成绩按课程名称的字符顺序输出
课程成绩输出格式:课程名称+英文空格+总成绩平均分
如果某门课程没有任何成绩信息,输出:课程名称+英文空格+"has no grades yet"
3)班级所有课程总成绩平均分按班级由低到高排序输出
格式:班级号+英文空格+总成绩平均分
如果某个班级没有任何成绩信息,输出:班级名称+英文空格+ "has no grades yet"
异常情况:
1)如果解析某个成绩信息时,课程名称不在已输入的课程列表中,输出:学号+英文空格+姓名+英文空格+":"+课程名称+英文空格+"does not exist"
2)如果解析某个成绩信息时,输入的成绩数量和课程的考核方式不匹配,输出:学号+英文空格+姓名+英文空格+": access mode mismatch"
以上两种情况如果同时出现,按第一种情况输出结果。
3)如果解析某个课程信息时,输入的课程性质和课程的考核方式不匹配,输出:课程名称+" : course type & access mode mismatch"
4)格式错误以及其他信息异常如成绩超出范围等,均按格式错误处理,输出"wrong format"
5)若出现重复的课程/成绩信息,只保留第一个课程信息,忽略后面输入的。
6)如果解析实验课程信息时,输入的分项成绩数量值和分项成绩权重的个数不匹配,输出:课程名称+" : number of scores does not match"
7)如果解析考试课、实验课时,分项成绩权重值的总和不等于1,输出:课程名称+" : weight value error"
信息约束:
1)成绩平均分只取整数部分,小数部分丢弃
参考类图(与第一次相同,其余内容自行补充):

输入样例1:
在这里给出一组输入。例如:
查看代码 java 实验 实验 4 0.2 0.3 0.2 0.3
end
输出样例1:
在这里给出相应的输出。例如:
查看代码 java has no grades yet
输入样例2:
在这里给出一组输入。例如:
查看代码 java 实验 实验 4 0.2 0.3 0.2
end
输出样例2:
在这里给出相应的输出。例如:
查看代码 java : number of scores does not match
输入样例3:
在这里给出一组输入。例如:
查看代码 java 实验 实验 4 0.2 0.3 0.2 0.1
end
输出样例3:
在这里给出相应的输出。例如:
查看代码 java : weight value error
输入样例4:
在这里给出一组输入。例如:
查看代码 java 实验 实验 4 0.2 0.3 0.2 0.3
20201116 张三 java 70 80 90 100
end
输出样例4:
在这里给出相应的输出。例如:
查看代码 20201116 张三 86
java 86
202011 86
输入样例5:
在这里给出一组输入。例如:
查看代码 java 实验 实验 4 0.2 0.3 0.2 0.3
20201116 张三 java 70 80 90 100 80
end
输出样例5:
在这里给出相应的输出。例如:
查看代码 20201116 张三 : access mode mismatch
20201116 张三 did not take any exams
java has no grades yet
202011 has no grades yet
设计与分析:
这题修改了类间关系,用分项成绩替换了原来的实验成绩考试成绩和考察成绩,而且也简化了一点输出,这道题相对于之前的迭代大改是在课程信息输入与处理,还有一部分是课程类学生类计算成绩的方法。考察成绩相当于一项的分项成绩,考试相当于两项,实验是多项,简化了成绩的处理,比原来的简单一点,最后在输出的时候也不用具体判断是哪一种成绩,直接输出对应分项成绩就行了。
类图:

SourceMonitor的生成报表内容:

采坑心得:
经过几次迭代,这次倒没什么大问题了,改完成绩类和课程类就差不多了。
改进建议:
最后还有两个测试点没过,不知道错在哪,可能要再多设置几个测试点才可以测出来,测试点也要多往提示上靠,尽量使功能完全。
源码如下:
查看代码 import java.util.Collections;
import java.util.Scanner;
import java.util.ArrayList;
import java.text.Collator;
import java.util.Comparator;
import java.util.Locale;
public class Main {
public static void main(String[] args) {
Scanner input = new Scanner(System.in);
Choosecourses choosecourses = new Choosecourses();
String courseMessage = input.nextLine();
while (!courseMessage.startsWith("end")){
if(checkCourseMessage(courseMessage)){
String[] message = courseMessage.split(" ");
Course course = new Course();
course.name = message[0];
if(!choosecourses.findCourse(course.name)){
if(message[1].equals("选修")){
if(message.length != 3){
if(message[2].equals("考试")){
if(message.length != 5){
System.out.println("wrong format");
}else {
course.way = "考试";
course.character = "必修";
course.courseScore = new CourseScore(2);
course.courseScore.score.weight[0] = Double.parseDouble(message[3]);//录入权重
course.courseScore.score.weight[1] = Double.parseDouble(message[4]);
if(course.courseScore.score.checkSubitem()){
choosecourses.getCourses().add(course);
} else {
System.out.println(message[0]+" : weight value error");
}
}
}else {
System.out.println(message[0]+" : course type & access mode mismatch");
}
}else if(message[2].equals("考察")){
course.way = message[2];
course.character = message[1];
course.courseScore = new CourseScore(1);
course.courseScore.score.weight[0] = 1;
choosecourses.getCourses().add(course);
}else {
System.out.println(message[0]+" : course type & access mode mismatch");
}
}else if((message[1].equals("必修")&& (message[2].equals("考察")||message[2].equals("实验")) ) ||
(message[1].equals("实验")&& (message[2].equals("考试")||message[2].equals("考察")) ) ){
System.out.println(message[0]+" : course type & access mode mismatch");
}else if(message[1].equals("必修")){
if(message.length != 5){//若必修但权重数量超过2
System.out.println("wrong format");
}else {
course.way = "考试";
course.character = "必修";
course.courseScore = new CourseScore(2);
course.courseScore.score.weight[0] = Double.parseDouble(message[3]);//录入权重
course.courseScore.score.weight[1] = Double.parseDouble(message[4]);
if(course.courseScore.score.checkSubitem()){
choosecourses.getCourses().add(course);
}
else {
System.out.println(message[0]+" : weight value error");
}
}
}else if(message[1].equals("实验")){
if(message.length != 4 + Integer.parseInt(message[3])){
System.out.println(message[0]+" : number of scores does not match");
}else {
course.way = "实验";
course.character = message[1];
course.courseScore = new CourseScore(Integer.parseInt(message[3]));
for (int i = 0;i < Integer.parseInt(message[3]);i++){//录入权重
course.courseScore.score.weight[i] = Double.parseDouble(message[4 + i]);
}
if(course.courseScore.score.checkSubitem()){
choosecourses.getCourses().add(course);
}
else {
System.out.println(message[0]+" : weight value error");
}
}
}
}
}else{
System.out.println("wrong format");
}
courseMessage = input.nextLine();
if(checkScoreMessage(courseMessage)){
break;
}
}
String scoreMessage = courseMessage;
while(!scoreMessage.startsWith("end")){
if(checkScoreMessage(scoreMessage)){
String[] message = scoreMessage.split(" ");
String classNumber = message[0].substring(0,6);
int flag = 0;
int flag1 = 0;
Course course1 = new Course();
for (Course course :choosecourses.getCourses()) {
if(course.name.equals(message[2])){
flag = 1;
if(message.length == 5&&course.way.equals("考试")){
course1.name = course.name;
course1.way = course.way;
course1.character = course.character;
course1.courseScore = new CourseScore(2);
for(int i = 0;i < 2;i++){
course1.courseScore.score.weight[i] = course.courseScore.score.weight[i];
}
course1.courseScore.score.scores[0] = Integer.parseInt(message[3]);
course1.courseScore.score.scores[1] = Integer.parseInt(message[4]);
flag1 = 2;
}else if(message.length == 4&&course.way.equals("考察")){
course1.name = course.name;
course1.way = course.way;
course1.character = course.character;
course1.courseScore = new CourseScore(1);
course1.courseScore.score.weight[0] = course.courseScore.score.weight[0];
course1.courseScore.score.scores[0] = Integer.parseInt(message[3]);
flag1 = 2;
}else if(message.length == 3 + course.courseScore.score.weight.length&&course.way.equals("实验")){
course1.name = course.name;
course1.way = course.way;
course1.character = course.character;
course1.courseScore = new CourseScore(course.courseScore.score.weight.length);
for(int i = 0;i < course.courseScore.score.weight.length;i++){
course1.courseScore.score.weight[i] = course.courseScore.score.weight[i];
}
for(int i = 3;i < message.length;i++){
course1.courseScore.score.scores[i - 3] = Integer.parseInt(message[i]);
}
flag1 = 2;
}else {
System.out.println(message[0]+" "+message[1]+" "+": access mode mismatch");
}
break;
}
}
if(choosecourses.findStudent(message[0]) == null){
Student student = new Student();
student.studentNumber = message[0];
student.name = message[1];
if(flag1 == 2){
student.courses.add(course1);
}
choosecourses.getStudent().add(student);
if(choosecourses.findClass(classNumber) == null){
Classes classes = new Classes(classNumber);
classes.students.add(student);
choosecourses.getClasses().add(classes);
}else{
choosecourses.findClass(classNumber).students.add(student);
}
}else if(choosecourses.findStudent(message[0]).findCourse(message[2]) == null&&flag1 == 2){
choosecourses.findStudent(message[0]).courses.add(course1);
}
if(flag == 0){
System.out.println(message[2]+" "+"does not exist");
}
}else {
System.out.println("wrong format");
}
scoreMessage = input.nextLine();
}
for(int i = 0;i < choosecourses.getStudent().size();i ++){
for(int j = 0;j < choosecourses.getStudent().size() - i - 1;j ++){
if(choosecourses.getStudent().get(j).studentNumber.compareTo(choosecourses.getStudent().get(j + 1).studentNumber)>0){
Student student = choosecourses.getStudent().get(j + 1);
choosecourses.getStudent().set(j + 1,choosecourses.getStudent().get(j));
choosecourses.getStudent().set(j,student);
}
}
}
for (Student student:choosecourses.getStudent()) {
if(student.courses.size() == 0){
System.out.println(student.studentNumber+" "+student.name+" did not take any exams");
}else{
System.out.println(student.studentNumber+" "+student.name+" "+student.getTotallAverage());
}
}
Collections.sort(choosecourses.getCourses());
for (Course course:choosecourses.getCourses()) {
if(choosecourses.findCourse1(course.name)){
System.out.println(course.name+" "+choosecourses.getCourseAverageTotallGrade(course));
}else{
System.out.println(course.name+" has no grades yet");
}
}
for(int i = 0;i < choosecourses.getClasses().size();i ++){
for(int j = 0;j < choosecourses.getClasses().size() - i - 1;j ++){
if(choosecourses.getClasses().get(j).number.compareTo(choosecourses.getClasses().get(j + 1).number)>0){
Classes classes = choosecourses.getClasses().get(j + 1);
choosecourses.getClasses().set(j + 1,choosecourses.getClasses().get(j));
choosecourses.getClasses().set(j,classes);
}
}
}
for (Classes classes:choosecourses.getClasses()) {
if(classes.isCourseEmpty()){
System.out.println(classes.number+" has no grades yet");
}else {
System.out.println(classes.number+" "+classes.getTotallAverage());
}
}
}
public static boolean checkCourseMessage(String courseMessage){
if(courseMessage.matches(".{1,11}\\s(必修|选修|实验)\\s(考试|考察|实验)\\s[456789](\\s0\\.[\\d])+")
||courseMessage.matches(".{1,11}\\s(必修|选修|实验)\\s(考试|考察|实验)")
||courseMessage.matches(".{1,11}\\s(必修|选修|实验)\\s(考试|考察|实验)(\\s0\\.[\\d]){2}")){
return true;
}
return false;
}
public static boolean checkScoreMessage(String scoreMessage){
if(scoreMessage.matches("[\\d]{8}\\s.{1,11}\\s.{1,11}(\\s([0-9]|[1-9][0-9]|100))+")
||scoreMessage.matches("[\\d]{8}\\s.{1,11}\\s.{1,11}\\s([0-9]|[1-9][0-9]|100)")
||scoreMessage.matches("[\\d]{8}\\s.{1,11}\\s.{1,11}\\s([0-9]|[1-9][0-9]|100)\\s([0-9]|[1-9][0-9]|100)"))
return true;
return false;
}
}
class Choosecourses {
private ArrayList<Course> courses = new ArrayList<>();
private ArrayList<Student> students = new ArrayList<>();
private ArrayList<CourseScore> scores = new ArrayList<>();
private ArrayList<Classes> classes = new ArrayList<>();
public ArrayList<Classes> getClasses() {
return classes;
}
public void setClasses(ArrayList<Classes> classes) {
this.classes = classes;
}
public ArrayList<Course> getCourses() {
return courses;
}
public void setCourses(ArrayList<Course> courses) {
this.courses = courses;
}
public ArrayList<Student> getStudent() {
return students;
}
public void setStudent(ArrayList<Student> student) {
this.students = student;
}
public Choosecourses() {
}
public int getCourseAverageTotallGrade(Course course){//单门课程平均总成绩
int sum = 0;
int i = 0;
for (Student student:students) {
for (Course course1: student.courses) {
if(course1.name.equals(course.name)){
sum += course1.getGrade();
i ++;
break;
}
}
}
if(i != 0)
return sum/i;
return 0;
}
public boolean findCourse(String str){
for (Course course: courses) {
if(course.name.equals(str)){
return true;
}
}
return false;
}
public boolean findCourse1(String course){
for (Student student: students) {
for (Course course1: student.courses) {
if(course1.name.equals(course)){
return true;
}
}
}
return false;
}
public Student findStudent(String str){
for (Student stu: students) {
if(stu.studentNumber.equals(str)){
return stu;
}
}
return null;
}
public Classes findClass(String str){
for (Classes class1: classes) {
if(class1.number.equals(str)){
return class1;
}
}
return null;
}
}
class Classes {
ArrayList<Student> students = new ArrayList<>();
String number;
public Classes() {
}
public Classes(String number) {
this.number = number;
}
public int getTotallAverage(){//班级总成绩平均分
int sum = 0;
for (Student student: students) {
sum += student.getTotallAverage();
}
return sum/students.size();
}
public boolean isCourseEmpty(){
for (Student stu: students) {
if(!stu.courses.isEmpty()){
return false;
}
}
return true;
}
}
class Course implements Comparable<Course>{
String name;
String character;
String way;
CourseScore courseScore;
public Course(int n) {
this.courseScore = new CourseScore(n);
}
public Course() {
}
public int getGrade(){
return courseScore.getTotall();
}
@Override
public int compareTo(Course course) {
Comparator<Object> comparator = Collator.getInstance(Locale.CHINA);
return ((Collator) comparator).compare(this.name,course.name); //升序
}
}
class CourseScore{
SubitemScore score;
public CourseScore(int n) {
this.score = new SubitemScore(n);
}
public int getTotall(){
return score.getTotall();
}
}
class Student {
String studentNumber;
String name;
ArrayList<Course> courses = new ArrayList<>();
public Student() {
}
public int getTotallAverage(){//所有课程总成绩平均分
int sum = 0;
for (Course course :courses) {
sum += course.getGrade();
}
if(courses.size() == 0)
return 0;
return sum/courses.size();
}
public Course findCourse(String name){
for (Course course :courses) {
if(course.name != null&&course.name.equals(name)){
return course;
}
}
return null;
}
}
class SubitemScore{
int[] scores;
double[] weight;
public SubitemScore(int n) {
this.scores = new int[n];
this.weight = new double[n];
}
public boolean checkSubitem(){
double sum = 0;
for (double w: weight) {
sum += w;
}
if(Math.abs(sum - 1) < 0.0000001){
return true;
}else {
return false;
}
}
public int getTotall(){//课程总分
double sum = 0;
for (int i = 0;i < scores.length;i++){
sum += scores[i] * weight[i];
}
return (int)sum;
}
}
- 总结
经过一学期的java学习,提升了自己的编程能力,同时也认识到还有很多要学的东西,自己的代码还有很多问题,功能不够完全,但是对于自己代码的查缺补漏往往是最头疼的,要多用调试功能查找问题完善代码,加强修改能力,自己的代码的设计也不够好,对于编程思想的实现也不完全,这些都要靠更多的练习。对于编程的态度也要多多认真,细心和严谨,增强逻辑思维能力。这一学期掌握了很多关于java的基础知识,了解认识了许多编程思想,但是学的还不够,希望以后也能多加自主学习。
- 意见和建议
还是有一部分上课讲的知识没有得到实践,这样就容易忘记,可以布置一些这方面的,题目可以要求多涉及一些知识点,这样平常多应用的话也会熟练一点不至于忘记。还可以加强对学生的监督,减少蒙混过关的情况,线上课的学习情况也不太好因为都不太会仔细看,也可以多加强这方面的监督。还有就是题目不用特别难,循序渐进,最后可以难一点,然后就是有时候因为平常的一些考试等情况写pta的时间会有点少,可以多了解这方面的信息来制定时间。
浙公网安备 33010602011771号