一、前言
1.知识点:总体来说涉及的知识点在之前都用过,所以没有太多值得说的。
LinkedHashMap和HashMap是有区别的,前者是按插入顺序储存元素,后者是哈希表
List ab = studenttotalscoremap.computeIfAbsent(grade.student, k -> Arrays.asList(0.0, 0.0));
ab.set(0, ab.get(0) + totalscore);这种方式可以计算指定位置的元素
其实主要还是逻辑问题,我所运用的知识点并没有太多
2.题量:不是太大,小题写的会快些,但是系列题找测试点的过程有时候很费时间。期末的选择题很多,而且很多都不会。
3.难度:适中,比起菜单系列真的好很多,部分测试点是真找不到。期末考试的题目由于给了类图也相对简单一些
二、设计与分析
7-3 课程成绩统计程序-2
课程成绩统计程序-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
实现代码:
1 package PTA8_2; 2 3 import java.util.*; 4 import java.util.regex.Matcher; 5 import java.util.regex.Pattern; 6 7 public class Main { 8 public static void main(String[] args) { 9 Scanner in = new Scanner(System.in); 10 GradeData gradeData = new GradeData(); 11 ArrayList<SelectedCourse> selectedCourses= new ArrayList<>(); 12 Pattern pattern = Pattern.compile("[0-9]*"); 13 String[] command = in.nextLine().split(" "); 14 while (!command[0].equals("end") && !pattern.matcher(command[0]).matches()){ 15 if(command[0].length() <= 10){ 16 if((command[1].equals("必修")||command[1].equals("选修")||command[1].equals("实验"))&&(command[2].equals("考试")||command[2].equals("考察")||command[2].equals("实验"))){ 17 // if(command[2].equals("实验")&&command.length < 5){ 18 // System.out.println("wrong format"); 19 // } 20 if(!gradeData.getCourse().contains(new Course(command[0],"",""))) 21 if(!gradeData.addCourse(command)) 22 System.out.println(command[0]+" : course type & access mode mismatch"); 23 } 24 else 25 System.out.println("wrong format"); 26 } 27 else 28 System.out.println("wrong format"); 29 command = in.nextLine().split(" "); 30 } 31 while (!command[0].equals("end")){ 32 if(command.length >= 4) { 33 SelectedCourse record = new SelectedCourse(command); 34 if (record.judge()) { 35 if (!selectedCourses.contains(record)) { 36 int judge = gradeData.addCourseScore(command); 37 if (judge == -1) 38 System.out.println(command[2] + " does not exist"); 39 else if (judge == -2) 40 System.out.println(command[0] + " " + command[1] + " : access mode mismatch"); 41 else if(judge == -3){ 42 command = in.nextLine().split(" "); 43 continue; 44 } 45 else if (judge == 0) 46 selectedCourses.add(record); 47 gradeData.addStudent(command,judge); 48 gradeData.addClass(command[0]); 49 } 50 } 51 else 52 System.out.println("wrong format"); 53 } 54 else 55 System.out.println("wrong format"); 56 command = in.nextLine().split(" "); 57 } 58 gradeData.studentSort(); 59 gradeData.courseSort(); 60 gradeData.classSort(); 61 gradeData.scorePrint(); 62 } 63 } 64 class GradeData { 65 private ArrayList<Course> courses = new ArrayList<>(); 66 private ArrayList<Student> students = new ArrayList<>(); 67 private ArrayList<Class> classes = new ArrayList<>(); 68 private Score score; 69 70 public GradeData() { 71 } 72 public ArrayList<Course> getCourse() { 73 return courses; 74 } 75 //添加课程信息 76 public boolean addCourse(String[] command){ 77 String course = command[0]; 78 String property = command[1]; 79 String form = command[2]; 80 if(course.length() > 10){ 81 System.out.println("wrong format"); 82 return true; 83 } 84 if(property.equals("必修") && !form.equals("考试")) 85 return false; 86 if(property.equals("选修") && !form.equals("考察")) 87 return false; 88 if(property.equals("实验") && !form.equals("实验")) 89 return false; 90 if(findCourse(course) >= 0) 91 return true; 92 if(findCourse(course) == -1){ 93 if(form.equals("实验")){ 94 int labnum = Integer.parseInt(command[3]); 95 double weights = 0.0; 96 Float[] weight = new Float[command.length - 4]; 97 if(labnum >= 4 && labnum <= 9){ 98 for(int i = 4; i < command.length; i++){ 99 weight[i-4] = Float.parseFloat(command[i]); 100 weights += Float.parseFloat(command[i]); 101 } 102 if(weight.length != labnum){ 103 System.out.println(course+" : number of scores does not match"); 104 return true; 105 } 106 else if(Math.abs(weights - 1) > 0.00001){ 107 System.out.println(course+" : weight value error"); 108 return true; 109 } 110 else{ 111 this.courses.add(new Course(course, property, form, labnum, weight)); 112 return true; 113 } 114 } 115 else { 116 System.out.println("wrong format"); 117 return true; 118 } 119 } 120 } 121 this.courses.add(new Course(course, property, form)); 122 return true; 123 } 124 public int addCourseScore(String[] command){ 125 String course = command[2]; 126 int fScore = 0; 127 int uScore = 0; 128 int index = findCourse(course); 129 if(index == -1){ 130 return -1; 131 } 132 Course replace = courses.get(index); 133 int totalScore = 0; 134 if(command.length == 4 && replace.form.equals("考察")){ 135 fScore = Integer.parseInt(command[3]); 136 ResearchScore score = new ResearchScore(fScore); 137 totalScore = score.getTotalScore(); 138 replace.addFinalScore(fScore); 139 } 140 else if(command.length == 5 && replace.form.equals("考试")) { 141 uScore = Integer.parseInt(command[3]); 142 fScore = Integer.parseInt(command[4]); 143 ExamScore score = new ExamScore(fScore,uScore); 144 totalScore = score.getTotalScore(); 145 replace.addFinalScore(fScore); 146 replace.addUsualScore(uScore); 147 } 148 else if(replace.form.equals("实验")){ 149 int labNum = replace.labnum; 150 if(labNum + 3 == command.length){ 151 LabScore score = new LabScore(command, replace.weight); 152 totalScore = score.getTotalScore(); 153 } 154 else 155 return -2; 156 } 157 else{ 158 return -2; 159 } 160 replace.addTotalScore(totalScore); 161 courses.set(index, replace); 162 return 0; 163 } 164 public int findCourse(String course){ 165 Course judge = new Course(course,"",""); 166 return courses.indexOf(judge); 167 } 168 public void addStudent(String[] command,int judge){ 169 int stuID = Integer.parseInt(command[0]); 170 String name = command[1]; 171 String course = command[2]; 172 int index = findStudent(stuID); 173 if(index == -1) 174 this.students.add(new Student(name, stuID)); 175 if(courses.contains(new Course(course,"",""))) { 176 String form = courses.get(findCourse(course)).form; 177 if(judge == 0) { 178 index = findStudent(stuID); 179 Student replace = students.get(index); 180 int totalScore = 0; 181 if (form.equals("考试")){ 182 ExamScore score = new ExamScore(Integer.parseInt(command[4]), Integer.parseInt(command[3])); 183 totalScore = score.getTotalScore(); 184 } 185 else if(form.equals("考察")){ 186 ResearchScore score = new ResearchScore(Integer.parseInt(command[3])); 187 totalScore = score.getTotalScore(); 188 } 189 else if(form.equals("实验")){ 190 Float[] weight = courses.get(findCourse(course)).weight; 191 LabScore score = new LabScore(command,weight); 192 totalScore = score.getTotalScore(); 193 } 194 replace.addTotalScore(totalScore); 195 students.set(index, replace); 196 } 197 } 198 } 199 public int findStudent(int stuID){ 200 Student judge = new Student("", stuID); 201 return students.indexOf(judge); 202 } 203 public void addClass(String stuID){ 204 if(stuID.length() == 8 && findClass(stuID.substring(0,6)) == -1) 205 classes.add(new Class(stuID.substring(0,6), -1)); 206 int index = findClass(stuID.substring(0,6)); 207 if(!students.get(findStudent(Integer.parseInt(stuID))).getTotalScores().isEmpty()){ 208 Class replace = classes.get(index); 209 replace.addStudent(students.get(findStudent(Integer.parseInt(stuID)))); 210 classes.set(index, replace); 211 } 212 } 213 public int findClass(String classNum){ 214 Class jugde = new Class(classNum, -1); 215 return classes.indexOf(jugde); 216 } 217 public void courseSort(){ 218 Collections.sort(courses, new Comparator<Course>() { 219 @Override 220 public int compare(Course o1, Course o2) { 221 Pattern p = Pattern.compile("[\u4e00-\u9fa5]"); 222 Matcher m1 = p.matcher(o1.course); 223 Matcher m2 = p.matcher(o2.course); 224 if(m1.find()&&m2.find()){ 225 if(o1.course.compareTo(o2.course) > 0){ 226 return 1; 227 }else if(o1.course.compareTo(o2.course) == 0){ 228 return 0; 229 }else if(o1.course.compareTo(o2.course) < 0){ 230 return -1; 231 } 232 } 233 else { 234 if(o1.course.compareTo(o2.course) > 0){ 235 return 1; 236 }else if(o1.course.compareTo(o2.course) == 0){ 237 return 0; 238 }else if(o1.course.compareTo(o2.course) < 0){ 239 return -1; 240 } 241 } 242 return 0; 243 } 244 }); 245 } 246 public void studentSort(){ 247 Collections.sort(students, new Comparator<Student>() { 248 @Override 249 public int compare(Student o1, Student o2) { 250 if(o1.getStuID() > o2.getStuID()){ 251 return 1; 252 }else if(o1.getStuID() == o2.getStuID()){ 253 return 0; 254 }else if(o1.getStuID() < o2.getStuID()){ 255 return -1; 256 } 257 return 0; 258 } 259 }); 260 } 261 public void classSort(){ 262 Collections.sort(classes, new Comparator<Class>() { 263 @Override 264 public int compare(Class o1, Class o2) { 265 if(o1.getClassNum().compareTo(o2.getClassNum()) > 0){ 266 return 1; 267 }else if(o1.getClassNum().compareTo(o2.getClassNum()) == 0){ 268 return 0; 269 }else if(o1.getClassNum().compareTo(o2.getClassNum()) < 0){ 270 return -1; 271 } 272 return 0; 273 } 274 }); 275 } 276 public void scorePrint(){ 277 for (int i = 0; i < students.size(); i++) 278 students.get(i).scoreShow(); 279 for (int i = 0; i < courses.size(); i++) 280 courses.get(i).scoreShow(); 281 for (int i = 0; i < classes.size(); i++) 282 classes.get(i).scoreShow(); 283 } 284 } 285 class SelectedCourse { 286 public Course course; 287 public Student student; 288 public String[] command; 289 public SelectedCourse(String[] command) { 290 this.course = new Course(command[2],"",""); 291 this.student = new Student("",Integer.parseInt(command[0])); 292 this.command = command; 293 } 294 public boolean judge(){ 295 Pattern pattern = Pattern.compile("0|[0-9]{1,2}|100"); 296 if(command[0].length() == 8 && command[1].length() <= 10){ 297 for(int i = 3; i < command.length; i++){ 298 if(!pattern.matcher(command[i]).matches()) 299 return false; 300 } 301 return true; 302 } 303 else 304 return false; 305 306 } 307 public boolean scorejudge(String[] command){ 308 Pattern pattern = Pattern.compile("0|[0-9]{1,2}|100"); 309 for(int i = 3; i < command.length; i++){ 310 if(pattern.matcher(command[i]).matches()) 311 return false; 312 } 313 return true; 314 } 315 public Course getCourse() { 316 return course; 317 } 318 public Student getStudent() { 319 return student; 320 } 321 @Override 322 public boolean equals(Object obj) { 323 SelectedCourse sc = (SelectedCourse) obj; 324 if(this.course.course.equals(sc.getCourse().course)&&this.student.getStuID() == sc.getStudent().getStuID()) 325 return true; 326 else 327 return false; 328 } 329 } 330 class Class { 331 private String classNum; 332 private int averageScore; 333 private ArrayList<Student> students = new ArrayList<>(); 334 public Class(String classNum, int averageScore) { 335 this.classNum = classNum; 336 this.averageScore = averageScore; 337 } 338 public String getClassNum() { 339 return classNum; 340 } 341 public void addStudent(Student student){ 342 students.add(student); 343 double score = 0.0; 344 for(int i = 0; i < students.size(); i++) 345 score += students.get(i).getAverageScore(); 346 this.averageScore = (int)(score / students.size()); 347 } 348 public int getAverageScore() { 349 return averageScore; 350 } 351 public void scoreShow(){ 352 if(students.isEmpty()) 353 System.out.println(classNum+" has no grades yet"); 354 else 355 System.out.println(classNum+" "+averageScore); 356 } 357 @Override 358 public boolean equals(Object obj) { 359 Class c = (Class) obj; 360 return this.classNum.equals(c.classNum); 361 } 362 } 363 class Student { 364 private String name; 365 private int stuID; 366 private ArrayList<Integer> totalScores = new ArrayList<>(); 367 public Student(String name, int stuID) { 368 this.name = name; 369 this.stuID = stuID; 370 } 371 public ArrayList<Integer> getTotalScores() { 372 return totalScores; 373 } 374 public int getStuID() { 375 return stuID; 376 } 377 public void addTotalScore(int score){ 378 totalScores.add(score); 379 } 380 public int getAverageScore() { 381 double score = 0.0; 382 for(int i = 0; i < totalScores.size(); i++) 383 score += totalScores.get(i); 384 if(totalScores.size() == 0) 385 return 0; 386 else 387 return (int)(score / totalScores.size()); 388 } 389 390 public void scoreShow(){ 391 if(totalScores.isEmpty()) 392 System.out.println(stuID+" "+name+" did not take any exams"); 393 else 394 System.out.println(stuID+" "+name+" "+getAverageScore()); 395 } 396 @Override 397 public boolean equals(Object obj) { 398 Student s = (Student)obj; 399 if(this.stuID == s.stuID) 400 return true; 401 else 402 return false; 403 } 404 } 405 406 class Course { 407 String course; 408 String property; 409 String form; 410 int labnum; 411 Float[] weight; 412 ArrayList<Integer> finalScores = new ArrayList<>(); 413 ArrayList<Integer> usualScores = new ArrayList<>(); 414 ArrayList<Integer> totalScores = new ArrayList<>(); 415 416 public Course(String course, String property, String form) { 417 this.course = course; 418 this.form = form; 419 this.property = property; 420 } 421 public Course(String course, String property, String form, int num, Float[] weight){ 422 this.course = course; 423 this.form = form; 424 this.property = property; 425 this.labnum = num; 426 this.weight = weight; 427 } 428 public void addFinalScore(int score){ 429 finalScores.add(score); 430 } 431 public void addUsualScore(int score){ 432 usualScores.add(score); 433 } 434 public void addTotalScore(int score){ 435 totalScores.add(score); 436 } 437 public int getAverageFinal(){ 438 double score = 0.0; 439 for(int i = 0; i < finalScores.size(); i++) 440 score += finalScores.get(i); 441 return (int)(score / finalScores.size()); 442 } 443 public int getAverageUsual(){ 444 double score = 0.0; 445 for(int i = 0; i < usualScores.size(); i++) 446 score += usualScores.get(i); 447 return (int)(score / usualScores.size()); 448 } 449 public int getAverageTotal(){ 450 double score = 0.0; 451 for(int i = 0; i < totalScores.size(); i++) 452 score += totalScores.get(i); 453 return (int)(score / totalScores.size()); 454 } 455 public void scoreShow(){ 456 if(totalScores.isEmpty()) 457 System.out.println(course+" has no grades yet"); 458 else{ 459 if(form.equals("考试")) 460 System.out.println(course+" "+getAverageUsual()+" "+getAverageFinal()+" "+getAverageTotal()); 461 else if(form.equals("考察")) 462 System.out.println(course+" "+getAverageFinal()+" "+getAverageTotal()); 463 else if(form.equals("实验")) 464 System.out.println(course+" "+getAverageTotal()); 465 } 466 } 467 @Override 468 public boolean equals(Object obj) { 469 Course c = (Course) obj; 470 return this.course.equals(c.course); 471 } 472 } 473 class Score { 474 int finalScore = 0; 475 int usualScore = 0; 476 ArrayList<Labgrade> labScores = new ArrayList<>(); 477 public Score() { 478 } 479 public Score(int finalCourse) { 480 this.finalScore = finalCourse; 481 } 482 public Score(int finalCourse, int usualCourse) { 483 this.finalScore = finalCourse; 484 this.usualScore = usualCourse; 485 } 486 } 487 class ExamScore{ 488 Score score; 489 public ExamScore(int finalCourse, int usualCourse) { 490 score = new Score(finalCourse ,usualCourse); 491 } 492 public int getTotalScore(){ 493 return (int)(score.finalScore * 0.7 + score.usualScore * 0.3); 494 } 495 } 496 497 class ResearchScore{ 498 Score score; 499 public ResearchScore(int finalCourse){ 500 score = new Score(finalCourse); 501 } 502 public int getTotalScore(){ 503 return score.finalScore; 504 } 505 } 506 class LabScore{ 507 Score score = new Score(); 508 public LabScore(String[] labscore,Float[] weight) { 509 for(int i = 3; i < labscore.length; i++) 510 score.labScores.add(new Labgrade(Integer.parseInt(labscore[i]), weight[i-3])); 511 } 512 public int getTotalScore(){ 513 double totalScore = 0.0; 514 for(Labgrade labgrade: score.labScores){ 515 totalScore += labgrade.score *labgrade.weight; 516 } 517 return (int)totalScore; 518 } 519 } 520 class Labgrade{ 521 int score; 522 double weight; 523 public Labgrade(int score, double weight) { 524 this.score = score; 525 this.weight = weight; 526 } 527 }
分析:
这道题类的设计国庆的时候写过1,我创建了成绩、课程、班级、学生、和选课类,成绩里面的属性时平时成绩、期末成绩、和总成绩,如果是考试的话就根据平时和期末成绩计算总成绩。课程类的话进行了构造函数,加进去了课程名称、课程性质、考核方式。学生类也进行了构造函数,加上了学号和学生姓名,但是我的班级类没有用上,其实按照现实情况来讲需要班级类,但是代码的实现并不需要,因为有更为方便的选课类,选课类是个神奇的东西,它把学生、课程、成绩一一对应组合到了一起,我把它存到了一个list集合grade,但在进行输出之前并未进行其他操作。
输入部分,我存储了课程信息,主要目的是在输入学生成绩的时候能即使判断是应按照考试还是考察计算学生成绩,再创建学生,最后把课程,成绩存到选课类里面。这就解决了没有错误信息处理的输入。
输出的话分3个部分,先是输出每个学生的平均分,并且需要排序,创建了 Map<Student, List>对象
studenttotalscoremap,student就指的是学生,list集合里面是他的所有科目成绩,但刚创建时肯定为空,这时候就遍历选课类的对象grade,以学生做键,把这个学生所有的成绩存到list里面,再遍历studenttotalscoremap输出。后面输出课程平均分和班级平均分几乎和上面的一样。这里面排序其实也是大问题,研究了很久,分别用三种方法给三个输出进行排序。
主要是这次研究了一下正则表达式,"(\S{1,10})( )(必修)( )(考察)")和"(\S{1,10})( )(必修|选修)( )(实验)"是错误的成绩信息,输出不匹配,"(\S{1,10})( )(((选修)( )(考试|考察))|((必修)(( )(考试))?))"是普通课程信息,"(\S{1,10})( )(实验)( )(实验)")是实验课程信息,"(\d{8})( )(\S{1,10})( )(\S{1,10})( )(\d{1,2}|(100))(( )(\d{1,2}|(100)))?"是考试考察的成绩信息,要是课程未找到输出不存在,再把把这个课程加进去,但是课程性质和考核方式都变成wrong,按长度判断是考试还是考察,如果课程是wrong的话总分就赋0,长度跟考核方式不匹配的话也赋0,并输出不匹配。"(\d{8})( )(\S{1,10})( )(\S{1,10})( )([4-9])( )((\d{1,2}|(100))( ))(\d{1,2}|(100))"这是实验成绩信息,要是课程未找到输出不存在,再把把这个课程加进去,但是课程性质和考核方式都变成wrong,长度不对也输出不匹配。
因为错误信息没写多少给我增添了不少负担,但是总的来说不算很难。
7-2 课程成绩统计程序-3
分数 64
作者 蔡轲
单位 南昌航空大学
课程成绩统计程序-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
代码如下:
1 package PTA8_2; 2 3 import java.util.*; 4 import java.util.regex.Matcher; 5 import java.util.regex.Pattern; 6 7 public class Main { 8 public static void main(String[] args) { 9 Scanner in = new Scanner(System.in); 10 GradeData gradeData = new GradeData(); 11 ArrayList<SelectedCourse> selectedCourses= new ArrayList<>(); 12 Pattern pattern = Pattern.compile("[0-9]*"); 13 String[] command = in.nextLine().split(" "); 14 while (!command[0].equals("end") && !pattern.matcher(command[0]).matches()){ 15 if(command[0].length() <= 10){ 16 if((command[1].equals("必修")||command[1].equals("选修")||command[1].equals("实验"))&&(command[2].equals("考试")||command[2].equals("考察")||command[2].equals("实验"))){ 17 // if(command[2].equals("实验")&&command.length < 5){ 18 // System.out.println("wrong format"); 19 // } 20 if(!gradeData.getCourse().contains(new Course(command[0],"",""))) 21 if(!gradeData.addCourse(command)) 22 System.out.println(command[0]+" : course type & access mode mismatch"); 23 } 24 else 25 System.out.println("wrong format"); 26 } 27 else 28 System.out.println("wrong format"); 29 command = in.nextLine().split(" "); 30 } 31 while (!command[0].equals("end")){ 32 if(command.length >= 4) { 33 SelectedCourse record = new SelectedCourse(command); 34 if (record.judge()) { 35 if (!selectedCourses.contains(record)) { 36 int judge = gradeData.addCourseScore(command); 37 if (judge == -1) 38 System.out.println(command[2] + " does not exist"); 39 else if (judge == -2) 40 System.out.println(command[0] + " " + command[1] + " : access mode mismatch"); 41 else if(judge == -3){ 42 command = in.nextLine().split(" "); 43 continue; 44 } 45 else if (judge == 0) 46 selectedCourses.add(record); 47 gradeData.addStudent(command,judge); 48 gradeData.addClass(command[0]); 49 } 50 } 51 else 52 System.out.println("wrong format"); 53 } 54 else 55 System.out.println("wrong format"); 56 command = in.nextLine().split(" "); 57 } 58 gradeData.studentSort(); 59 gradeData.courseSort(); 60 gradeData.classSort(); 61 gradeData.scorePrint(); 62 } 63 } 64 class GradeData { 65 private ArrayList<Course> courses = new ArrayList<>(); 66 private ArrayList<Student> students = new ArrayList<>(); 67 private ArrayList<Class> classes = new ArrayList<>(); 68 private Score score; 69 70 public GradeData() { 71 } 72 public ArrayList<Course> getCourse() { 73 return courses; 74 } 75 //添加课程信息 76 public boolean addCourse(String[] command){ 77 String course = command[0]; 78 String property = command[1]; 79 String form = command[2]; 80 if(course.length() > 10){ 81 System.out.println("wrong format"); 82 return true; 83 } 84 if(property.equals("必修") && !form.equals("考试")) 85 return false; 86 if(property.equals("选修") && !form.equals("考察")) 87 return false; 88 if(property.equals("实验") && !form.equals("实验")) 89 return false; 90 if(findCourse(course) >= 0) 91 return true; 92 if(findCourse(course) == -1){ 93 if(form.equals("实验")){ 94 int labnum = Integer.parseInt(command[3]); 95 double weights = 0.0; 96 Float[] weight = new Float[command.length - 4]; 97 if(labnum >= 4 && labnum <= 9){ 98 for(int i = 4; i < command.length; i++){ 99 weight[i-4] = Float.parseFloat(command[i]); 100 weights += Float.parseFloat(command[i]); 101 } 102 if(weight.length != labnum){ 103 System.out.println(course+" : number of scores does not match"); 104 return true; 105 } 106 else if(Math.abs(weights - 1) > 0.00001){ 107 System.out.println(course+" : weight value error"); 108 return true; 109 } 110 else{ 111 this.courses.add(new Course(course, property, form, labnum, weight)); 112 return true; 113 } 114 } 115 else { 116 System.out.println("wrong format"); 117 return true; 118 } 119 } 120 } 121 this.courses.add(new Course(course, property, form)); 122 return true; 123 } 124 public int addCourseScore(String[] command){ 125 String course = command[2]; 126 int fScore = 0; 127 int uScore = 0; 128 int index = findCourse(course); 129 if(index == -1){ 130 return -1; 131 } 132 Course replace = courses.get(index); 133 int totalScore = 0; 134 if(command.length == 4 && replace.form.equals("考察")){ 135 fScore = Integer.parseInt(command[3]); 136 ResearchScore score = new ResearchScore(fScore); 137 totalScore = score.getTotalScore(); 138 replace.addFinalScore(fScore); 139 } 140 else if(command.length == 5 && replace.form.equals("考试")) { 141 uScore = Integer.parseInt(command[3]); 142 fScore = Integer.parseInt(command[4]); 143 ExamScore score = new ExamScore(fScore,uScore); 144 totalScore = score.getTotalScore(); 145 replace.addFinalScore(fScore); 146 replace.addUsualScore(uScore); 147 } 148 else if(replace.form.equals("实验")){ 149 int labNum = replace.labnum; 150 if(labNum + 3 == command.length){ 151 LabScore score = new LabScore(command, replace.weight); 152 totalScore = score.getTotalScore(); 153 } 154 else 155 return -2; 156 } 157 else{ 158 return -2; 159 } 160 replace.addTotalScore(totalScore); 161 courses.set(index, replace); 162 return 0; 163 } 164 public int findCourse(String course){ 165 Course judge = new Course(course,"",""); 166 return courses.indexOf(judge); 167 } 168 public void addStudent(String[] command,int judge){ 169 int stuID = Integer.parseInt(command[0]); 170 String name = command[1]; 171 String course = command[2]; 172 int index = findStudent(stuID); 173 if(index == -1) 174 this.students.add(new Student(name, stuID)); 175 if(courses.contains(new Course(course,"",""))) { 176 String form = courses.get(findCourse(course)).form; 177 if(judge == 0) { 178 index = findStudent(stuID); 179 Student replace = students.get(index); 180 int totalScore = 0; 181 if (form.equals("考试")){ 182 ExamScore score = new ExamScore(Integer.parseInt(command[4]), Integer.parseInt(command[3])); 183 totalScore = score.getTotalScore(); 184 } 185 else if(form.equals("考察")){ 186 ResearchScore score = new ResearchScore(Integer.parseInt(command[3])); 187 totalScore = score.getTotalScore(); 188 } 189 else if(form.equals("实验")){ 190 Float[] weight = courses.get(findCourse(course)).weight; 191 LabScore score = new LabScore(command,weight); 192 totalScore = score.getTotalScore(); 193 } 194 replace.addTotalScore(totalScore); 195 students.set(index, replace); 196 } 197 } 198 } 199 public int findStudent(int stuID){ 200 Student judge = new Student("", stuID); 201 return students.indexOf(judge); 202 } 203 public void addClass(String stuID){ 204 if(stuID.length() == 8 && findClass(stuID.substring(0,6)) == -1) 205 classes.add(new Class(stuID.substring(0,6), -1)); 206 int index = findClass(stuID.substring(0,6)); 207 if(!students.get(findStudent(Integer.parseInt(stuID))).getTotalScores().isEmpty()){ 208 Class replace = classes.get(index); 209 replace.addStudent(students.get(findStudent(Integer.parseInt(stuID)))); 210 classes.set(index, replace); 211 } 212 } 213 public int findClass(String classNum){ 214 Class jugde = new Class(classNum, -1); 215 return classes.indexOf(jugde); 216 } 217 public void courseSort(){ 218 Collections.sort(courses, new Comparator<Course>() { 219 @Override 220 public int compare(Course o1, Course o2) { 221 Pattern p = Pattern.compile("[\u4e00-\u9fa5]"); 222 Matcher m1 = p.matcher(o1.course); 223 Matcher m2 = p.matcher(o2.course); 224 if(m1.find()&&m2.find()){ 225 if(o1.course.compareTo(o2.course) > 0){ 226 return 1; 227 }else if(o1.course.compareTo(o2.course) == 0){ 228 return 0; 229 }else if(o1.course.compareTo(o2.course) < 0){ 230 return -1; 231 } 232 } 233 else { 234 if(o1.course.compareTo(o2.course) > 0){ 235 return 1; 236 }else if(o1.course.compareTo(o2.course) == 0){ 237 return 0; 238 }else if(o1.course.compareTo(o2.course) < 0){ 239 return -1; 240 } 241 } 242 return 0; 243 } 244 }); 245 } 246 public void studentSort(){ 247 Collections.sort(students, new Comparator<Student>() { 248 @Override 249 public int compare(Student o1, Student o2) { 250 if(o1.getStuID() > o2.getStuID()){ 251 return 1; 252 }else if(o1.getStuID() == o2.getStuID()){ 253 return 0; 254 }else if(o1.getStuID() < o2.getStuID()){ 255 return -1; 256 } 257 return 0; 258 } 259 }); 260 } 261 public void classSort(){ 262 Collections.sort(classes, new Comparator<Class>() { 263 @Override 264 public int compare(Class o1, Class o2) { 265 if(o1.getClassNum().compareTo(o2.getClassNum()) > 0){ 266 return 1; 267 }else if(o1.getClassNum().compareTo(o2.getClassNum()) == 0){ 268 return 0; 269 }else if(o1.getClassNum().compareTo(o2.getClassNum()) < 0){ 270 return -1; 271 } 272 return 0; 273 } 274 }); 275 } 276 public void scorePrint(){ 277 for (int i = 0; i < students.size(); i++) 278 students.get(i).scoreShow(); 279 for (int i = 0; i < courses.size(); i++) 280 courses.get(i).scoreShow(); 281 for (int i = 0; i < classes.size(); i++) 282 classes.get(i).scoreShow(); 283 } 284 } 285 class SelectedCourse { 286 public Course course; 287 public Student student; 288 public String[] command; 289 public SelectedCourse(String[] command) { 290 this.course = new Course(command[2],"",""); 291 this.student = new Student("",Integer.parseInt(command[0])); 292 this.command = command; 293 } 294 public boolean judge(){ 295 Pattern pattern = Pattern.compile("0|[0-9]{1,2}|100"); 296 if(command[0].length() == 8 && command[1].length() <= 10){ 297 for(int i = 3; i < command.length; i++){ 298 if(!pattern.matcher(command[i]).matches()) 299 return false; 300 } 301 return true; 302 } 303 else 304 return false; 305 306 } 307 public boolean scorejudge(String[] command){ 308 Pattern pattern = Pattern.compile("0|[0-9]{1,2}|100"); 309 for(int i = 3; i < command.length; i++){ 310 if(pattern.matcher(command[i]).matches()) 311 return false; 312 } 313 return true; 314 } 315 public Course getCourse() { 316 return course; 317 } 318 public Student getStudent() { 319 return student; 320 } 321 @Override 322 public boolean equals(Object obj) { 323 SelectedCourse sc = (SelectedCourse) obj; 324 if(this.course.course.equals(sc.getCourse().course)&&this.student.getStuID() == sc.getStudent().getStuID()) 325 return true; 326 else 327 return false; 328 } 329 } 330 class Class { 331 private String classNum; 332 private int averageScore; 333 private ArrayList<Student> students = new ArrayList<>(); 334 public Class(String classNum, int averageScore) { 335 this.classNum = classNum; 336 this.averageScore = averageScore; 337 } 338 public String getClassNum() { 339 return classNum; 340 } 341 public void addStudent(Student student){ 342 students.add(student); 343 double score = 0.0; 344 for(int i = 0; i < students.size(); i++) 345 score += students.get(i).getAverageScore(); 346 this.averageScore = (int)(score / students.size()); 347 } 348 public int getAverageScore() { 349 return averageScore; 350 } 351 public void scoreShow(){ 352 if(students.isEmpty()) 353 System.out.println(classNum+" has no grades yet"); 354 else 355 System.out.println(classNum+" "+averageScore); 356 } 357 @Override 358 public boolean equals(Object obj) { 359 Class c = (Class) obj; 360 return this.classNum.equals(c.classNum); 361 } 362 } 363 class Student { 364 private String name; 365 private int stuID; 366 private ArrayList<Integer> totalScores = new ArrayList<>(); 367 public Student(String name, int stuID) { 368 this.name = name; 369 this.stuID = stuID; 370 } 371 public ArrayList<Integer> getTotalScores() { 372 return totalScores; 373 } 374 public int getStuID() { 375 return stuID; 376 } 377 public void addTotalScore(int score){ 378 totalScores.add(score); 379 } 380 public int getAverageScore() { 381 double score = 0.0; 382 for(int i = 0; i < totalScores.size(); i++) 383 score += totalScores.get(i); 384 if(totalScores.size() == 0) 385 return 0; 386 else 387 return (int)(score / totalScores.size()); 388 } 389 390 public void scoreShow(){ 391 if(totalScores.isEmpty()) 392 System.out.println(stuID+" "+name+" did not take any exams"); 393 else 394 System.out.println(stuID+" "+name+" "+getAverageScore()); 395 } 396 @Override 397 public boolean equals(Object obj) { 398 Student s = (Student)obj; 399 if(this.stuID == s.stuID) 400 return true; 401 else 402 return false; 403 } 404 } 405 406 class Course { 407 String course; 408 String property; 409 String form; 410 int labnum; 411 Float[] weight; 412 ArrayList<Integer> finalScores = new ArrayList<>(); 413 ArrayList<Integer> usualScores = new ArrayList<>(); 414 ArrayList<Integer> totalScores = new ArrayList<>(); 415 416 public Course(String course, String property, String form) { 417 this.course = course; 418 this.form = form; 419 this.property = property; 420 } 421 public Course(String course, String property, String form, int num, Float[] weight){ 422 this.course = course; 423 this.form = form; 424 this.property = property; 425 this.labnum = num; 426 this.weight = weight; 427 } 428 public void addFinalScore(int score){ 429 finalScores.add(score); 430 } 431 public void addUsualScore(int score){ 432 usualScores.add(score); 433 } 434 public void addTotalScore(int score){ 435 totalScores.add(score); 436 } 437 public int getAverageFinal(){ 438 double score = 0.0; 439 for(int i = 0; i < finalScores.size(); i++) 440 score += finalScores.get(i); 441 return (int)(score / finalScores.size()); 442 } 443 public int getAverageUsual(){ 444 double score = 0.0; 445 for(int i = 0; i < usualScores.size(); i++) 446 score += usualScores.get(i); 447 return (int)(score / usualScores.size()); 448 } 449 public int getAverageTotal(){ 450 double score = 0.0; 451 for(int i = 0; i < totalScores.size(); i++) 452 score += totalScores.get(i); 453 return (int)(score / totalScores.size()); 454 } 455 public void scoreShow(){ 456 if(totalScores.isEmpty()) 457 System.out.println(course+" has no grades yet"); 458 else{ 459 if(form.equals("考试")) 460 System.out.println(course+" "+getAverageUsual()+" "+getAverageFinal()+" "+getAverageTotal()); 461 else if(form.equals("考察")) 462 System.out.println(course+" "+getAverageFinal()+" "+getAverageTotal()); 463 else if(form.equals("实验")) 464 System.out.println(course+" "+getAverageTotal()); 465 } 466 } 467 @Override 468 public boolean equals(Object obj) { 469 Course c = (Course) obj; 470 return this.course.equals(c.course); 471 } 472 } 473 class Score { 474 int finalScore = 0; 475 int usualScore = 0; 476 ArrayList<Labgrade> labScores = new ArrayList<>(); 477 public Score() { 478 } 479 public Score(int finalCourse) { 480 this.finalScore = finalCourse; 481 } 482 public Score(int finalCourse, int usualCourse) { 483 this.finalScore = finalCourse; 484 this.usualScore = usualCourse; 485 } 486 } 487 class ExamScore{ 488 Score score; 489 public ExamScore(int finalCourse, int usualCourse) { 490 score = new Score(finalCourse ,usualCourse); 491 } 492 public int getTotalScore(){ 493 return (int)(score.finalScore * 0.7 + score.usualScore * 0.3); 494 } 495 } 496 497 class ResearchScore{ 498 Score score; 499 public ResearchScore(int finalCourse){ 500 score = new Score(finalCourse); 501 } 502 public int getTotalScore(){ 503 return score.finalScore; 504 } 505 } 506 class LabScore{ 507 Score score = new Score(); 508 public LabScore(String[] labscore,Float[] weight) { 509 for(int i = 3; i < labscore.length; i++) 510 score.labScores.add(new Labgrade(Integer.parseInt(labscore[i]), weight[i-3])); 511 } 512 public int getTotalScore(){ 513 double totalScore = 0.0; 514 for(Labgrade labgrade: score.labScores){ 515 totalScore += labgrade.score *labgrade.weight; 516 } 517 return (int)totalScore; 518 } 519 } 520 class Labgrade{ 521 int score; 522 double weight; 523 public Labgrade(int score, double weight) { 524 this.score = score; 525 this.weight = weight; 526 } 527 }
分析:
这次改动的有点多,且最终也没有拿到满分。课程类增添记录权重的数组。
输入的第三个是不是考试考察或者实验,这仨是课程信息,是考试时,长度不为5或者第二个或者权重相加不为1或者是实验就报错,把权重存到数组,若没问题就加到map里面。考察没啥东西就不说了,实验的话,次数不是4-9或者数量不匹配或者加权不得1都报错,没问题加到map。
"(\d{8})( )(\S{1,10})( )(\S{1,10})( )((\d{1,2}|(100))( ))(\d{1,2}|(100))"用于是判断成绩信息的,就是把上次判断是不是实验的4-9扣去了,首先还是判断有没有课,没课就加上,但课程性质和考核方式都是wrong,权重数组是null,再判断是不是wrong,是的话就输出不存在,这样就避免了上次代码只能输出一次的弊端。接下来判断是什么课,考核方式是实验或者wrong就是实验课,如果是正常信息的话就用一个for循环让每项成绩跟权重相乘得到总分,考试考察的话跟上次大致一样。但是最后还是有两个测试点没有过去TT
期末考试
(期末怎么可以有选择题啊怎么可以有选择题啊怎么可以有选择题啊怎么可以有选择题啊啊啊啊啊啊啊)
7-1 立体图形问题 分数 10 作者 段喜龙 单位 南昌航空大学编程求得正方体和正三棱锥的表面积和体积,要求必须体现扩展性(继承)和多态性。
类结构如下图所示(参考):
试编程完成如上类设计,主方法源码如下(可直接拷贝使用):
public static void main(String[] args) {
// TODO Auto-generated method stub
Scanner input = new Scanner(System.in);
double side = input.nextDouble();
display(new Cube(side));
display(new RegularPyramid(side));
}
其中,display(Solid solid)
方法为定义在Main类中的静态方法,作用为体现程序的多态性。
注:正三棱锥的体积计算公式为底面积*高/3。
输入格式:
输入一个实型数,分别作为正方体的边长和正三棱锥的边长。
输出格式:
分别输出正方体的表面积、体积以及正棱锥的表面积和体积。保留两位小数,建议使用String.format(“%.2f”,value)
进行小数位数控制。
输入样例:
在这里给出一组输入。例如:
2.5
输出样例:
在这里给出相应的输出。例如:
37.50
15.63
10.83
1.84
代码如下:
1 import java.util.Scanner; 2 3 // 父类:固体 4 abstract class Solid { 5 // 获取表面积 6 public abstract double getArea(); 7 // 获取体积 8 public abstract double getVolume(); 9 } 10 11 // 子类:正方体 12 class Cube extends Solid { 13 private double side; 14 public Cube(double side) { 15 this.side = side; 16 } 17 // 重写父类方法:获取表面积 18 @Override 19 public double getArea() { 20 return 6 * side * side; 21 } 22 23 // 重写父类方法:获取体积 24 @Override 25 public double getVolume() { 26 return side * side * side; 27 } 28 } 29 30 // 子类:正三棱锥 31 class RegularPyramid extends Solid { 32 private double side; 33 private double height; 34 35 public RegularPyramid(double side, double height) { 36 this.side = side; 37 this.height = height; 38 } 39 40 // 重写父类方法:获取表面积 41 @Override 42 public double getArea() { 43 return Math.sqrt(3) * side * side ; 44 } 45 46 // 重写父类方法:获取体积 47 @Override 48 public double getVolume() { 49 double baseArea = (side * side * Math.sqrt(3)) / 4; 50 return baseArea * height / 3; 51 } 52 } 53 54 public class Main { 55 public static void main(String[] args) { 56 Scanner input = new Scanner(System.in); 57 double side = input.nextDouble(); 58 double height = (side * Math.sqrt(6)) / 3; 59 60 61 display(new Cube(side)); 62 display(new RegularPyramid(side, height)); 63 } 64 65 // 多态方法:展示固体的表面积和体积 66 public static void display(Solid solid) { 67 System.out.printf("%.2f\n", solid.getArea()); 68 System.out.printf("%.2f\n", solid.getVolume()); 69 } 70 }
分析:这个题目真的蛮简单的,就是一开始忘记了正三棱锥的高怎么求。。。感觉更多就像是考察数学,没有什么很难的部分,更多是为了后面的题目铺垫。
7-2 魔方问题
分数 20
作者 段喜龙
单位 南昌航空大学
问题描述:本问题中的魔方有两种,一种是正方体魔方,一种是正三棱锥魔方,其中,正方体或正三棱锥魔方是由单元正方体或正三棱锥组成,单元正方体或正三棱锥的个数由阶数(即层数)决定,即魔方边长=阶数*单元边长。魔方如下图所示:
利用“立体图形”问题源码,实现如下功能:
魔方有三个属性:颜色,阶数,类型(正方体魔方、正三棱锥魔方),程序要求输出魔方的颜色、表面积和体积。参考设计类图如下所示:
主方法部分可参考如下源码(可拷贝直接使用):
public class Main {
public static void main(String[] args) {
// TODO Auto-generated method stub
Scanner input = new Scanner(System.in);
String color = input.next();
int layer = input.nextInt();
double side = input.nextDouble();
RubikCube cube1 = new SquareCube(color, layer,new Cube(side));
color = input.next();
layer = input.nextInt();
side = input.nextDouble();
RubikCube cube2 = new RegularPyramidCube(color, layer,new RegularPyramid(side));
display(cube1);
display(cube2);
}
}
其中,display(RubikCube cube)
方法为Main类中定义的静态方法,用户输出魔方的信息,用于体现多态性。
输入格式:
第一部分:正方体魔方颜色、阶数、单元正方体边长,以空格或回车分隔;
第二部分:正三棱锥魔方颜色、阶数、单元正三棱锥边长,以空格或回车分隔。
输出格式:
正方体魔方颜色
正方体魔方表面积
正方体魔方体积
正三棱锥魔方颜色
正三棱锥魔方表面积
正三棱锥魔方体积
注:小数点保留两位
输入样例:
在这里给出一组输入。例如:
red 3 4.5
black 4 2.1
输出样例:
在这里给出相应的输出。例如:
red
1093.50
2460.38
black
122.21
69.85
代码如下:
1 import java.util.Scanner; 2 3 abstract class RubikCube { 4 protected String color; 5 protected int layer; 6 protected String type; 7 8 public RubikCube(String color, int layer, String type) { 9 this.color = color; 10 this.layer = layer; 11 this.type = type; 12 } 13 14 public abstract double calculateSurfaceArea(); 15 public abstract double calculateVolume(); 16 } 17 18 class Cube extends RubikCube { 19 private double side; 20 21 public Cube(String color, int layer, double side) { 22 super(color, layer, "正方体"); 23 this.side = side; 24 } 25 26 @Override 27 public double calculateSurfaceArea() { 28 return 6 * layer * layer * side * side; 29 } 30 31 @Override 32 public double calculateVolume() { 33 return layer * layer * layer * side * side * side; 34 } 35 } 36 37 class RegularPyramid extends RubikCube { 38 private double side; 39 40 public RegularPyramid(String color, int layer, double side) { 41 super(color, layer, "正三棱锥"); 42 this.side = side; 43 } 44 45 @Override 46 public double calculateSurfaceArea() { 47 return Math.sqrt(3) * layer * side * layer * side ; 48 } 49 50 @Override 51 public double calculateVolume() { 52 double height = (layer * side * Math.sqrt(6)) / 3; 53 double baseArea = (layer * side *layer * side * Math.sqrt(3)) / 4; 54 return baseArea * height / 3; 55 } 56 } 57 58 class SquareCube extends Cube { 59 public SquareCube(String color, int layer, double side) { 60 super(color, layer, side); 61 } 62 } 63 64 class RegularPyramidCube extends RegularPyramid { 65 public RegularPyramidCube(String color, int layer, double side) { 66 super(color, layer, side); 67 } 68 } 69 70 public class Main { 71 public static void display(RubikCube cube) { 72 System.out.println(cube.color); 73 System.out.printf("%.2f%n", cube.calculateSurfaceArea()); 74 System.out.printf("%.2f%n", cube.calculateVolume()); 75 } 76 77 public static void main(String[] args) { 78 Scanner input = new Scanner(System.in); 79 80 String color = input.next(); 81 int layer = input.nextInt(); 82 double side = input.nextDouble(); 83 84 RubikCube cube1 = new SquareCube(color, layer, side); 85 86 color = input.next(); 87 layer = input.nextInt(); 88 side = input.nextDouble(); 89 90 RubikCube cube2 = new RegularPyramidCube(color, layer, side); 91 display(cube1); 92 display(cube2); 93 } 94 }
分析:这个也很简单不是很难,一次就过了。
7-3 魔方排序问题 分数 20 作者 段喜龙 单位 南昌航空大学在魔方问题的基础上,重构类设计,实现列表内魔方的排序功能(按照魔方的体积进行排序)。
提示:题目中RubikCube类要实现Comparable接口。
其中,Main类源码如下(可直接拷贝使用):
public class Main {
public static void main(String[] args) {
// TODO Auto-generated method stub
Scanner input = new Scanner(System.in);
String color;
int layer;
double side;
RubikCube cube;
ArrayList<RubikCube> list = new ArrayList<>();
int choice = input.nextInt();
while(choice != 0) {
switch(choice) {
case 1://SquareCube
color = input.next();
layer = input.nextInt();
side = input.nextDouble();
cube = new SquareCube(color, layer,new Cube(side));
list.add(cube);
break;
case 2://RegularPyramidCube
color = input.next();
layer = input.nextInt();
side = input.nextDouble();
cube = new RegularPyramidCube(color, layer,new RegularPyramid(side));
list.add(cube);
break;
}
choice = input.nextInt();
}
list.sort(Comparator.naturalOrder());//正向排序
for(int i = 0; i < list.size(); i++) {
System.out.print(list.get(i).getColor() + " " +
String.format("%.2f", list.get(i).getArea()) + " " +
String.format("%.2f", list.get(i).getVolume()) );
System.out.println("");
}
}
}
输入格式:
输入魔方类型(1:正方体魔方;2:正三棱锥魔方;0:结束输入)
魔方颜色、魔方阶数、魔方单元正方体、正三棱锥边长
..循环..
输出格式:
按魔方体积升序输出列表中各魔方的信息(实型数均保留两位小数),输出样式参见输出样例。
输入样例:
在这里给出一组输入。例如:
1 blue 3 4.5
2 red 4 2.1
1 yellow 5 2.3
2 black 4 9.42
1 white 4 5.4423
0
输出样例:
在这里给出相应的输出。例如:
red 122.21 69.85
yellow 793.50 1520.88
blue 1093.50 2460.38
black 2459.14 6304.73
white 2843.39 10316.38
代码如下:
1 import java.util.ArrayList; 2 import java.util.Collections; 3 import java.util.Comparator; 4 import java.util.Scanner; 5 6 abstract class RubikCube implements Comparable<RubikCube> { 7 protected String color; 8 protected int layer; 9 protected String type; 10 11 public RubikCube(String color, int layer, String type) { 12 this.color = color; 13 this.layer = layer; 14 this.type = type; 15 } 16 17 public abstract double calculateSurfaceArea(); 18 public abstract double calculateVolume(); 19 20 public String getColor() { 21 return color; 22 } 23 24 public double getArea() { 25 return calculateSurfaceArea(); 26 } 27 28 public double getVolume() { 29 return calculateVolume(); 30 } 31 32 @Override 33 public int compareTo(RubikCube other) { 34 double thisVolume = calculateVolume(); 35 double otherVolume = other.calculateVolume(); 36 if (thisVolume < otherVolume) { 37 return -1; 38 } else if (thisVolume > otherVolume) { 39 return 1; 40 } else { 41 return 0; 42 } 43 } 44 } 45 46 class Cube extends RubikCube { 47 private double side; 48 49 public Cube(String color, int layer, double side) { 50 super(color, layer, "正方体"); 51 this.side = side; 52 } 53 54 @Override 55 public double calculateSurfaceArea() { 56 return 6 * layer * layer * side * side; 57 } 58 59 @Override 60 public double calculateVolume() { 61 return layer * layer * layer * side * side * side; 62 } 63 } 64 65 class RegularPyramid extends RubikCube { 66 private double side; 67 68 public RegularPyramid(String color, int layer, double side) { 69 super(color, layer, "正三棱锥"); 70 this.side = side; 71 } 72 73 @Override 74 public double calculateSurfaceArea() { 75 return Math.sqrt(3) * layer * side * layer * side ; 76 } 77 78 @Override 79 public double calculateVolume() { 80 double height = (layer * side * Math.sqrt(6)) / 3; 81 double baseArea = (layer * side *layer * side * Math.sqrt(3)) / 4; 82 return baseArea * height / 3; 83 } 84 } 85 86 class SquareCube extends Cube { 87 public SquareCube(String color, int layer, double side) { 88 super(color, layer, side); 89 } 90 } 91 92 class RegularPyramidCube extends RegularPyramid { 93 public RegularPyramidCube(String color, int layer, double side) { 94 super(color, layer, side); 95 } 96 } 97 98 public class Main { 99 public static void main(String[] args) { 100 Scanner input = new Scanner(System.in); 101 102 String color; 103 int layer; 104 double side; 105 RubikCube cube; 106 107 ArrayList<RubikCube> list = new ArrayList<>(); 108 109 int choice = input.nextInt(); 110 111 while (choice != 0) { 112 switch (choice) { 113 case 1:// SquareCube 114 color = input.next(); 115 layer = input.nextInt(); 116 side = input.nextDouble(); 117 cube = new SquareCube(color, layer, side); 118 list.add(cube); 119 break; 120 case 2:// RegularPyramidCube 121 color = input.next(); 122 layer = input.nextInt(); 123 side = input.nextDouble(); 124 cube = new RegularPyramidCube(color, layer, side); 125 list.add(cube); 126 break; 127 } 128 choice = input.nextInt(); 129 } 130 131 Collections.sort(list);// 使用自然排序(按体积升序) 132 133 for (RubikCube rubikCube : list) { 134 System.out.println( 135 rubikCube.getColor() + " " + String.format("%.2f", rubikCube.getArea()) + " " + String.format("%.2f", rubikCube.getVolume())); 136 } 137 } 138 }
分析:
这题主要加了排序的功能,还好之前运用过Collections.sort()函数,其他的都主要在前期代码上面迭代,也不算很难。7-4 销售步枪问题(附加题) 分数 10 作者 段喜龙 单位 南昌航空大学
前亚利桑那州境内的一位步枪销售商销售密苏里州制造的步枪机(lock)、枪托(stock)和枪管(barrel)。枪机卖45美元,枪托卖30美元,枪管卖25美元。销售商每月至少要售出一支完整的步枪,且生产限额是销售商在一个月内可销售70个枪机、80个枪托和90个枪管。
根据每个月的销售情况,计算销售商的佣金(提成)算法如下:
-
不到(含)1000美元的部分为10%;
-
1000(含)~1800美元的部分为15%;
-
超过1800美元的部分为20%。
佣金程序生成月份销售报告,汇总销售商的销售总额和佣金。
编程要求:必须符合面向对象编程,且保证类设计的单一职责模式,使用面向过程编程判定0分。
提示:可以设置一个销售订单类。参考类图如下:
输入格式:
输入销售商每个月售出枪机、枪托、枪管的数量,可以用空格或者回车分隔。
输出格式:
分别输出销售商在该月的销售额和佣金,中间用空格分开。
输入样例1:
在这里给出一组输入。例如:
30 40 50
输出样例1:
在这里给出相应的输出。例如:
3800.00 620.00
输入样例2:
在这里给出一组输入。例如:
88 56 98
输出样例2:
在这里给出相应的输出。例如:
Wrong Format
代码:
1 import java.util.Scanner; 2 // 销售订单类 3 class SalesOrder { 4 private int rifleQuantity; // 步枪数量 5 private int stockQuantity; // 枪托数量 6 private int barrelQuantity; // 枪管数量 7 8 // 构造函数,用于初始化销售订单 9 public SalesOrder(int rifleQuantity, int stockQuantity, int barrelQuantity) { 10 this.rifleQuantity = rifleQuantity; 11 this.stockQuantity = stockQuantity; 12 this.barrelQuantity = barrelQuantity; 13 } 14 15 // 获取步枪数量 16 public int getRifleQuantity() { 17 return rifleQuantity; 18 } 19 20 // 获取枪托数量 21 public int getStockQuantity() { 22 return stockQuantity; 23 } 24 25 // 获取枪管数量 26 public int getBarrelQuantity() { 27 return barrelQuantity; 28 } 29 } 30 31 // 步枪类 32 class Rifle { 33 private double riflePrice; // 步枪价格 34 35 // 构造函数,用于初始化步枪价格 36 public Rifle(double riflePrice) { 37 this.riflePrice = riflePrice; 38 } 39 40 // 获取步枪价格 41 public double getRiflePrice() { 42 return riflePrice; 43 } 44 } 45 46 // 销售商类 47 class Salesman { 48 private Rifle rifle; // 步枪对象 49 private double stockPrice; // 枪托价格 50 private double barrelPrice; // 枪管价格 51 52 // 构造函数,用于初始化销售商的步枪、枪托和枪管价格 53 public Salesman(Rifle rifle, double stockPrice, double barrelPrice) { 54 this.rifle = rifle; 55 this.stockPrice = stockPrice; 56 this.barrelPrice = barrelPrice; 57 } 58 59 // 计算销售商的销售额 60 public double calculateTotalSales(SalesOrder salesOrder) { 61 int rifleQuantity = salesOrder.getRifleQuantity(); 62 int stockQuantity = salesOrder.getStockQuantity(); 63 int barrelQuantity = salesOrder.getBarrelQuantity(); 64 65 double rifleSales = rifle.getRiflePrice() * rifleQuantity; 66 double stockSales = stockPrice * stockQuantity; 67 double barrelSales = barrelPrice * barrelQuantity; 68 69 return rifleSales + stockSales + barrelSales; 70 } 71 72 // 计算销售商的佣金 73 public double calculateCommission(double totalSales) { 74 double commission; 75 if (totalSales <= 1000) { 76 commission = totalSales * 0.1; 77 } else if (totalSales <= 1800) { 78 commission = 1000 * 0.1 + (totalSales - 1000) * 0.15; 79 } else { 80 commission = 1000 * 0.1 + 800 * 0.15 + (totalSales - 1800) * 0.2; 81 } 82 return commission; 83 } 84 } 85 86 // 主程序 87 public class Main { 88 public static void main(String[] args) { 89 // 创建步枪对象 90 Rifle rifle = new Rifle(45.0); 91 92 // 创建销售商对象 93 Salesman salesman = new Salesman(rifle, 30.0, 25.0); 94 95 // 读取输入 96 Scanner scanner = new Scanner(System.in); 97 int rifleQuantity = scanner.nextInt(); 98 int stockQuantity = scanner.nextInt(); 99 int barrelQuantity = scanner.nextInt(); 100 scanner.close(); 101 102 //判断是否卖出一把完整步枪内 103 if(rifleQuantity < 1 || stockQuantity < 1 || barrelQuantity < 1){ 104 System.out.printf("Wrong Format"); 105 return; 106 } 107 //判断是否在生产限额内 108 if(rifleQuantity > 70 || stockQuantity > 80 || barrelQuantity > 90){ 109 System.out.printf("Wrong Format"); 110 return; 111 } 112 // 创建销售订单对象 113 SalesOrder salesOrder = new SalesOrder(rifleQuantity, stockQuantity, barrelQuantity); 114 115 // 计算销售额和佣金 116 double totalSales = salesman.calculateTotalSales(salesOrder); 117 double commission = salesman.calculateCommission(totalSales); 118 119 // 输出结果 120 System.out.printf("%.2f %.2f\n", totalSales, commission); 121 } 122 }
分析:经过大作业的捶打,这种题目都比较得心应手。除了在输入合法性的时候漏考虑了一个点,其他的都很快通过。
三、踩坑心得
- 应该多给代码标上备注。比如这几次题目集相隔时间有点远,导致我对之前的代码没什么记忆,又没有注释,让我读起来还需要仔细回忆一下这些东西都是干什么的,浪费了很多时间,也感到了注释的重要性。
- 不要瞎写,先动脑子再动键盘,我往往是感觉哪不对就改,并没有想好怎么改,导致代码冗余。
- 最近写大作业发现打断点去调试真的很有用会比自己想想想好很多。
四、改进意见
1.日后不能只追求写出来,更要运用好一些上课的设计模式,去灵活构造,而不是做底层码农。
2.与那些算法设计的比较好的同学交流交流,询问他们是如何想到更加优质的代码和更加良好的结构的。
五、总结
按照类图去做题、构造真的让我感受到了面向对象的很多优势,每次迭代需要改的地方都很少,java最重要的地方还是逻辑和涉及,这次是老师为我们想好类图,但下次我们就要自己想了,在写代码之前缕清题目逻辑并确定一个合适的类与类的关系是至关重要的。
好文要顶 关注我 收藏该文 标签:return,String,int,BLOG,score,new,public From: https://www.cnblogs.com/YiKe-QUAN/p/17891953.html