一、前言:
这是第二次的blog,接下来关于这最近三次的PTA大作业,只有第一次是上次答题判题程序的延续,接下来则是一个全新的关于电路的设计,最新的电路设计相较于之前的答题判题程序来说的话,难度确实有所下降。前两次中都含有三道题,而最后一次的PTA则是删去了其余两道题,让我们专心钻研新的电路设计。而关于第四次的PTA大作业,第二三题依旧是对于我们基础的训练,而由于第一题依旧是前三次的答题判题的迭代,这就导致我没有办法去很好的完成它,同时也是我很后悔的一个点,但是段老师为了照顾我们,特意让蔡老师设计了一个新的,也就是第五、六次的第一题,让我们可以重新开始设计,不要再踩之前设计的欠缺的坑了。后两次虽然难度相较于之前要小了,但是对于我来说也是一个不小的挑战。
二、设计与分析:
PTA第四次作业:
关于PTA的第四次大作业,第二题是要求我们设计一个学生类和它的一个子类——本科生类,这道题目呢主要是考察我们对于继承的理解和运用,所以并不是很复杂,只是一个较为常见的继承,所以这里我们就简单看一下题目和代码就行了。
题目描述:
设计一个学生类(Student)和它的一个子类——本科生类(Undergraduate)。要求如下:
(1)Student类有姓名(name)和年龄(age)属性,两者的访问权限为protected;一个包含两个参数的构造方法,用于给姓名和年龄属性赋值;一个show( )方法用于输出Student的属性信息,输出格式为Student[name=XXX,age=XX]。
(2)Undergraduate类增加一个专业(major)属性,该属性的访问权限为private;有一个包含三个参数的构造方法,前两个参数用于给继承的年龄和姓名属性赋值,第三个参数给专业属性赋值;一个show( )方法用于输出Undergraduate的属性信息,输出格式为Undergraduate[name=XXX,age=XXX,major=XXX]。
在测试类Main的main( )方法中,调用Student类的带参数的构造方法创建对象object1,调用Undergraduate类的带参数的构造方法创建对象object2,然后分别调用它们的show( )方法。
输入格式:
第一行从键盘输入姓名(字符串)、年龄(整数)给Student类的对象object1,中间以空格隔开。
第二行从键盘输入姓名(字符串)、年龄(整数)、专业(字符串)给Undergraduate类的对象object2,中间以空格隔开。
输出格式:
第一行输出object1的姓名、年龄,格式为:Student[name=XXX,age=XX]
第二行输出object2的姓名、年龄、专业,格式为: Undergraduate[name=XXX,age=XXX,major=XXX]
输入样例:
在这里给出一组输入。例如:
张三 12
李四 19 计算机科学与技术
输出样例:
在这里给出相应的输出,其中的标点符号为英文格式。例如:
Student[name=张三,age=12]
Undergraduate[name=李四,age=19,major=计算机科学与技术]
接下来是我的一个代码实现,因为这题比较简单我就不放类图了。
`import java.util.Scanner;
class Student {
protected String name;
protected int age;
public Student(String name, int age) {
this.name = name;
this.age = age;
}
public void show() {
System.out.println("Student[name=" + name + ",age=" + age + "]");
}
}
class Undergraduate extends Student {
private String major;
public Undergraduate(String name, int age, String major) {
super(name, age);
this.major = major;
}
public void show() {
System.out.println("Undergraduate[name=" + name + ",age=" + age + ",major=" + major + "]");
}
}
public class Main {
public static void main(String[] args) {
Scanner scanner = new Scanner(System.in);
String name1 = scanner.next();
int age1 = scanner.nextInt();
Student object1 = new Student(name1, age1);
String name2 = scanner.next();
int age2 = scanner.nextInt();
String major = scanner.next();
Undergraduate object2 = new Undergraduate(name2, age2, major);
object1.show();
object2.show();
scanner.close();
}
}`
接下来是第四次PTA的第一题的题面,相较于上次增加了输入选择题题目信息、输入填空题题目信息、输出顺序变化、多张试卷信息等方面:
设计实现答题程序,模拟一个小型的测试,要求输入题目信息、试卷信息、答题信息、学生信息、删除题目信息,根据输入题目信息中的标准答案判断答题的结果。本题在答题判题程序-3基础上新增的内容统一附加在输出格式说明之后,用粗体标明。
输入格式:
程序输入信息分五种,信息可能会打乱顺序混合输入。
1、题目信息
题目信息为独行输入,一行为一道题,多道题可分多行输入。
格式:"#N:"+题目编号+" "+"#Q:"+题目内容+" "#A:"+标准答案
格式约束:
1、题目的输入顺序与题号不相关,不一定按题号顺序从小到大输入。
2、允许题目编号有缺失,例如:所有输入的题号为1、2、5,缺少其中的3号题。此种情况视为正常。
样例:#N:1 #Q:1+1= #A:2
#N:2 #Q:2+2= #A:4
2、试卷信息
试卷信息为独行输入,一行为一张试卷,多张卷可分多行输入数据。 \
格式:"#T:"+试卷号+" "+题目编号+"-"+题目分值+" "+题目编号+"-"+题目分值+...
格式约束:
题目编号应与题目信息中的编号对应。
一行信息中可有多项题目编号与分值。
样例:#T:1 3-5 4-8 5-2
3、学生信息
学生信息只输入一行,一行中包括所有学生的信息,每个学生的信息包括学号和姓名,格式如下。
格式:"#X:"+学号+" "+姓名+"-"+学号+" "+姓名....+"-"+学号+" "+姓名
格式约束:
答案数量可以不等于试卷信息中题目的数量,没有答案的题目计0分,多余的答案直接忽略,答案之间以英文空格分隔。
样例:
#S:1 #A:5 #A:22
1是试卷号
5是1号试卷的顺序第1题的题目答案
4、答卷信息
答卷信息按行输入,每一行为一张答卷的答案,每组答案包含某个试卷信息中的题目的解题答案,答案的顺序号与试 卷信息中的题目顺序相对应。答卷中:
格式:"#S:"+试卷号+" "+学号+" "+"#A:"+试卷题目的顺序号+"-"+答案内容+...
格式约束:
答案数量可以不等于试卷信息中题目的数量,没有答案的题目计0分,多余的答案直接忽略,答案之间以英文空格分隔。
答案内容可以为空,即””。
答案内容中如果首尾有多余的空格,应去除后再进行判断。
答卷信息中仅包含试卷号、学号,而没有后续内容的,视为一张空白卷,为有效信息,不做格式错误处理。
样例:
#T:1 1-5 3-2 2-5 6-9 4-10 7-3
#S:1 20201103 #A:2-5 #A:6-4
1是试卷号
20201103是学号
2-5中的2是试卷中顺序号,5是试卷第2题的答案,即T中3-2的答案
6-4中的6是试卷中顺序号,4是试卷第6题的答案,即T中7-3的答案
注意:不要混淆顺序号与题号
5、删除题目信息
删除题目信息为独行输入,每一行为一条删除信息,多条删除信息可分多行输入。该信息用于删除一道题目信息,题目被删除之后,引用该题目的试卷依然有效,但被删除的题目将以0分计,同时在输出答案时,题目内容与答案改为一条失效提示,例如:”the question 2 invalid~0”
格式:"#D:N-"+题目号
格式约束:
题目号与第一项”题目信息”中的题号相对应,不是试卷中的题目顺序号。
本题暂不考虑删除的题号不存在的情况。
样例:
N:1 #Q:1+1= #A:2
N:2 #Q:2+2= #A:4
T:1 1-5 2-8
X:20201103 Tom-20201104 Jack
S:1 20201103 #A:1-5 #A:2-4
D:N-2
end
输出:
alert: full score of test paper1 is not 100 points
1+1=5false
the question 2 invalid~0
20201103 Tom: 0 0~0
答题信息以一行"end"标记结束,"end"之后的信息忽略。
输出格式:
1、试卷总分警示
该部分仅当一张试卷的总分分值不等于100分时作提示之用,试卷依然属于正常试卷,可用于后面的答题。如果总分等于100 分,该部分忽略,不输出。
格式:"alert: full score of test paper"+试卷号+" is not 100 points"
约束:有多张试卷时,按输入信息的先后顺序输出警示。
样例:alert: full score of test paper2 is not 100 points
2、答卷信息
一行为一道题的答题信息,根据试卷的题目的数量输出多行数据。
格式:题目内容+""+答案++""+判题结果(true/false)
约束:如果输入的答案信息少于试卷的题目数量,每一个缺失答案的题目都要输出"answer is null" 。
样例:
answer is null
3+2=~5~true
4+6=~22~false.
answer is null
3、判分信息
判分信息为一行数据,是一条答题记录所对应试卷的每道小题的计分以及总分,计分输出的先后顺序与题目题号相对应。
格式:学号+" "+姓名+": "+题目得分+" "+....+题目得分+"~"+总分
格式约束:
1、没有输入答案的题目、被删除的题目、答案错误的题目计0分
2、判题信息的顺序与输入答题信息中的顺序相同
样例:20201103 Tom: 0 0~0
根据输入的答卷的数量以上2、3项答卷信息与判分信息将重复输出。
4、被删除的题目提示信息
当某题目被试卷引用,同时被删除时,答案中输出提示信息。样例见第5种输入信息“删除题目信息”。
5、题目引用错误提示信息
试卷错误地引用了一道不存在题号的试题,在输出学生答案时,提示”non-existent question~”加答案。例如:
输入:
N:1 #Q:1+1= #A:2
T:1 3-8
X:20201103 Tom-20201104 Jack-20201105 Www
S:1 20201103 #A:1-4
end
输出:
alert: full score of test paper1 is not 100 points
non-existent question~0
20201103 Tom: 0~0
如果答案输出时,一道题目同时出现答案不存在、引用错误题号、题目被删除,只提示一种信息,答案不存在的优先级最高,例如:
输入:
N:1 #Q:1+1= #A:2
T:1 3-8
X:20201103 Tom-20201104 Jack-20201105 Www
S:1 20201103
end
输出:
alert: full score of test paper1 is not 100 points
answer is null
20201103 Tom: 0~0
6、格式错误提示信息
输入信息只要不符合格式要求,均输出”wrong format:”+信息内容。
例如:wrong format:2 #Q:2+2= #4
7、试卷号引用错误提示输出
如果答卷信息中试卷的编号找不到,则输出”the test paper number does not exist”,答卷中的答案不用输出,参见样例8。
8、学号引用错误提示信息
如果答卷中的学号信息不在学生列表中,答案照常输出,判分时提示错误。参见样例9。
本次作业新增内容:
1、输入选择题题目信息
题目信息为独行输入,一行为一道题,多道题可分多行输入。
格式:"#Z:"+题目编号+" "+"#Q:"+题目内容+" "#A:"+标准答案
格式基本的约束与一般的题目输入信息一致。
新增约束:标准答案中如果包含多个正确答案(多选题),正确答案之间用英文空格分隔。
例如:
Z:2 #Q:宋代书法有苏黄米蔡四家,分别是: #A:苏轼 黄庭坚 米芾 蔡襄
多选题输出:
输出格式与一般答卷题目的输出一致,判断结果除了true、false,增加一项”partially correct”表示部分正确。
多选题给分方式:
答案包含所有正确答案且不含错误答案给满分;包含一个错误答案或完全没有答案给0分;包含部分正确答案且不含错误答案给一半分,如果一半分值为小数,按截尾规则只保留整数部分。
例如:
N:1 #Q:1+1= #A:2
Z:2 #Q:党十八大报告提出要加强()建设。A 政务诚信 B 商务诚信 C社会诚信 D司法公信 #A:A B C D
T:1 1-5 2-9
X:20201103 Tom
S:1 20201103 #A:1-5 #A:2-A C
end
输出:
alert: full score of test paper1 is not 100 points
1+1=5false
党十八大报告提出要加强()建设。A 政务诚信 B 商务诚信 C社会诚信 D司法公信~A C~partially correct
20201103 Tom: 0 4~4
2、输入填空题题目信息
题目信息为独行输入,一行为一道题,多道题可分多行输入。
格式:"#K:"+题目编号+" "+"#Q:"+题目内容+" "#A:"+标准答案
格式基本的约束与一般的题目输入信息一致。
例如:#K:2 #Q:古琴在古代被称为: #A:瑶琴或七弦琴
填空题输出:
输出格式与一般答卷题目的输出一致,判断结果除了true、false,增加一项”partially correct”表示部分正确。
填空题给分方式:
答案与标准答案内容完全匹配给满分,包含一个错误字符或完全没有答案给0分,包含部分正确答案且不含错误字符给一半分,如果一半分值为小数,按截尾规则只保留整数部分。
例如:
N:1 #Q:1+1= #A:2
K:2 #Q:古琴在古代被称为: #A:瑶琴或七弦琴
T:1 1-5 2-10
X:20201103 Tom
S:1 20201103 #A:1-5 #A:2-瑶琴
end
输出:
alert: full score of test paper1 is not 100 points
1+1=5false
古琴在古代被称为:瑶琴partially correct
20201103 Tom: 0 5~5
3、输出顺序变化
只要是正确格式的信息,可以以任意的先后顺序输入各类不同的信息。比如试卷可以出现在题目之前,删除题目的信息可以出现在题目之前等。
例如:
T:1 1-5 2-10
N:1 #Q:1+1= #A:2
K:2 #Q:古琴在古代被称为: #A:瑶琴或七弦琴
X:20201103 Tom
S:1 20201103 #A:1-5 #A:2-古筝
end
输出:
alert: full score of test paper1 is not 100 points
1+1=5false
古琴在古代被称为:古筝false
20201103 Tom: 0 0~0
4、多张试卷信息
本题考虑多个同学有多张不同试卷的答卷的情况。输出顺序优先级为学号、试卷号,按从小到大的顺序先按学号排序,再按试卷号。
例如:
T:1 1-5 2-10
T:2 1-8 2-21
N:1 #Q:1+1= #A:2
S:2 20201103 #A:1-2 #A:2-古筝
S:1 20201103 #A:1-5 #A:2-瑶琴或七弦琴
S:1 20201104 #A:1-2 #A:2-瑟
S:2 20201104 #A:1-5 #A:2-七弦琴
X:20201103 Tom-20201104 Jack
K:2 #Q:古琴在古代被称为: #A:瑶琴或七弦琴
end
输出:
alert: full score of test paper1 is not 100 points
alert: full score of test paper2 is not 100 points
1+1=5false
古琴在古代被称为:瑶琴或七弦琴true
20201103 Tom: 0 10~10
1+1=2true
古琴在古代被称为:古筝false
20201103 Tom: 8 0~8
1+1=2true
古琴在古代被称为:瑟false
20201104 Jack: 5 0~5
1+1=5false
古琴在古代被称为:七弦琴partially correct
20201104 Jack: 0 10~10
新增的题目异常情况的处理与一般题目相同,具体样例参考上一次大作业的样例说明:
答题判题程序-3题面.pdf
输入样例1:
多选题测试,不含删除。例如:
N:1 #Q:1+1= #A:2
Z:2 #Q:党十八大报告提出要加强()建设。A 政务诚信 B 商务诚信 C社会诚信 D司法公信 #A:A B C D
T:1 1-5 2-9
X:20201103 Tom
S:1 20201103 #A:1-5 #A:2-A C
end
输出样例1:
在这里给出相应的输出。例如:
alert: full score of test paper1 is not 100 points
1+1=5false
党十八大报告提出要加强()建设。A 政务诚信 B 商务诚信 C社会诚信 D司法公信~A C~partially correct
20201103 Tom: 0 4~4
输入样例2:
填空题测试,不含删除。例如:
N:1 #Q:1+1= #A:2
K:2 #Q:古琴在古代被称为: #A:瑶琴或七弦琴
T:1 1-5 2-10
X:20201103 Tom
S:1 20201103 #A:1-5 #A:2-瑶琴
end
输出样例2:
在这里给出相应的输出。例如:
alert: full score of test paper1 is not 100 points
1+1=5false
古琴在古代被称为:瑶琴partially correct
20201103 Tom: 0 5~5
输入样例3:
乱序测试,不含删除。例如:
T:1 1-5 2-10
N:1 #Q:1+1= #A:2
K:2 #Q:古琴在古代被称为: #A:瑶琴或七弦琴
X:20201103 Tom
S:1 20201103 #A:1-5 #A:2-古筝
end
输出样例3:
在这里给出相应的输出。例如:
alert: full score of test paper1 is not 100 points
1+1=5false
古琴在古代被称为:古筝false
20201103 Tom: 0 0~0
输入样例4:
两个同学多张不同试卷的答卷,不含删除。例如:
T:1 1-5 2-10
T:2 1-8 2-21
N:1 #Q:1+1= #A:2
S:2 20201103 #A:1-2 #A:2-古筝
S:1 20201104 #A:1-2 #A:2-瑟
S:1 20201103 #A:1-5 #A:2-瑶琴或七弦琴
S:2 20201104 #A:1-5 #A:2-七弦琴
X:20201103 Tom-20201104 Jack
K:2 #Q:古琴在古代被称为: #A:瑶琴或七弦琴
end
输出样例4:
在这里给出相应的输出。例如:
alert: full score of test paper1 is not 100 points
alert: full score of test paper2 is not 100 points
1+1=5false
古琴在古代被称为:瑶琴或七弦琴true
20201103 Tom: 0 10~10
1+1=2true
古琴在古代被称为:古筝false
20201103 Tom: 8 0~8
1+1=2true
古琴在古代被称为:瑟false
20201104 Jack: 5 0~5
1+1=5false
古琴在古代被称为:七弦琴partially correct
20201104 Jack: 0 10~10
点击查看代码
import java.util.*;
import java.util.regex.*;
import java.util.Scanner;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import java.util.ArrayList;
import java.util.List;
class Question {
private String content;
private String answer;
public Question(String content, String answer) {
this.content = content;
this.answer = answer;
}
public String getContent() {
return content;
}
public String getAnswer() {
return answer;
}
}
class TestPaper {
private int paperNumber;
private Map<Integer, Question> questions = new HashMap<>();
private Map<Integer, Integer> questionScores = new HashMap<>();
public TestPaper(int paperNumber) {
this.paperNumber = paperNumber;
}
public void addQuestion(int number, String content, String answer, int score) {
questions.put(number, new Question(content, answer));
questionScores.put(number, score);
}
public int getPaperNumber() {
return paperNumber;
}
public Map<Integer, Question> getQuestions() {
return questions;
}
public Map<Integer, Integer> getQuestionScores() {
return questionScores;
}
}
class Student {
private String id;
private String name;
// 其他字段和方法
public Student(String id, String name) {
this.id = id;
this.name = name;
}
// getter 和 setter 方法
public String getId() {
return id;
}
public void setId(String id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
class AnswerSheet {
private TestPaper testPaper;
private String[] answers;
private boolean[] judgments;
public AnswerSheet(TestPaper testPaper) {
this.testPaper = testPaper;
this.answers = new String[testPaper.getQuestionScores().size()];
this.judgments = new boolean[testPaper.getQuestionScores().size()];
}
public void addAnswer(String answer, int index) {
answers[index] = answer;
}
public void judgeAnswers() {
for (int i = 0; i < testPaper.getQuestionScores().size(); i++) {
String correctAnswer = testPaper.getQuestions().get(i + 1).getAnswer();
judgments[i] = answers[i] != null && answers[i].equals(correctAnswer);
}
}
public void outputResults(String studentId, String studentName) {
int totalScore = 0;
for (int score : testPaper.getQuestionScores().values()) {
totalScore += score;
}
if (totalScore != 100) {
System.out.println("alert: full score of test paper" + testPaper.getPaperNumber() + " is not 100 points");
}
for (int i = 0; i < answers.length; i++) {
Question question= testPaper.getQuestions().get(i + 1);
if (question != null) {
String content = question.getContent(); // 获取题目内容
String answer = (answers[i] != null ? answers[i] : "answer is null");
String judgment = String.valueOf(judgments[i]);
String result = content + "~" + answer + "~" + judgment;
System.out.println(result); // 输出题目内容、学生答案、判断结果
} else {
System.out.println("the question " + (i + 1) + " invalid ~ 0");
}
}
int totalStudentScore = 0;
for (int i = 0; i < testPaper.getQuestionScores().size(); i++) {
if (judgments[i]) {
totalStudentScore += testPaper.getQuestionScores().get(i + 1);
}
}
// 输出学生的得分情况
System.out.println(studentId + " " + studentName + ": " + formatStudentScore() + "~" + totalStudentScore);
}
private String formatStudentScore() {
StringBuilder sb = new StringBuilder();
for (int i = 0; i < testPaper.getQuestionScores().size(); i++) {
sb.append(judgments[i] ? testPaper.getQuestionScores().get(i + 1) : "0").append(" ");
}
return sb.toString().trim();
}
}
public class Main {
public static void main(String[] args) {
Scanner scanner = new Scanner(System.in);
TestPaper testPaper = null;
Map<String, Student> studentMap = new HashMap<>();
String studentId = "";
String studentName = "";
while (true) {
String line = scanner.nextLine();
if (line.equals("end")) {
break;
}
Matcher matcher;
if ((matcher = Pattern.compile("#N:(\\d+) #Q:(.*?) #A:(\\d+)").matcher(line)).matches()) {
// Parse #N: section
int number = Integer.parseInt(matcher.group(1));
String questionContent = matcher.group(2); // 获取题目内容
String answer = matcher.group(3);
testPaper = new TestPaper(number);
testPaper.addQuestion(number, questionContent, answer, 5); // 将题目内容添加到试卷中
} else if ((matcher = Pattern.compile("#T:(\\d+) (\\d+-\\d+)").matcher(line)).matches()) {
// Parse #T: section
int paperNumber = Integer.parseInt(matcher.group(1));
String[] parts = matcher.group(2).split("-");
int questionNumber = Integer.parseInt(parts[0]);
int score = Integer.parseInt(parts[1]);
testPaper.addQuestion(questionNumber, "", "", score);
} else if ((matcher = Pattern.compile("#X:(\\d+) (.+)").matcher(line)).matches()) {
// Parse #X: section
studentId = matcher.group(1);
studentName = matcher.group(2);
} else if ((matcher = Pattern.compile("#S:(\\d+) (\\d+) #A:(\\d+-\\d+)").matcher(line)).matches()) {
// Parse #S: section
if (testPaper != null) {
AnswerSheet answerSheet = new AnswerSheet(testPaper);
String[] parts = matcher.group(3).split("-");
int questionNumber = Integer.parseInt(parts[0]);
String answer = parts[1];
answerSheet.addAnswer(answer, questionNumber - 1);
answerSheet.judgeAnswers();
answerSheet.outputResults(studentId, studentName);
} else {
System.out.println("Test paper not initialized!");
}
}
}
}
}
上面是我的代码,下面是我的类图:
关于这道题目,相信通过我的类图也能看出我前几次设计的问题,从根本上也就是设计上出现了缺陷和漏洞,也就是我的类设计中类太少了,每一个类都承担了太多的职责,没法达到单一职责不说,还造成了每一个类都十分的繁琐和冗长,这也就使得在检查测试点时,我根本没法进行有效的修改从而通过测试点,更别提效率问题了。然而这道题目又不仅仅是简单的设计,它已经无法改变,所以我只能硬着头皮写下去,导致我也就不出意外的没有及格。
第五次PTA:
首先我们看一下他的其他两道题目,大同小异,都是通过修改老师给的代码从而实现功能,由于较为基础,所以这里我们随便挑选一道题目进行展示:
阅读程序,按照题目需求修改程序
功能需求:
使用集合存储3个员工的信息(有序);
通过迭代器依次找出所有的员工。
提示:学生复制以下代码到编程区,并按需求进行调试修改。
// 1、导入相关包
//定义员工类
class Employee {
private String name;
private int age;
public Employee() {
super();
}
public Employee(String name, int age) {
super();
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
}
//主函数
public class Main {
public static void main(String[] args) {
// 1、创建有序集合对象
Collection c ;
// 创建3个员工元素对象
for (int i = 0; i < 3; i++) {
Scanner sc = new Scanner(System.in);
String employeeName = sc.nextLine();
int employeeAge = sc.nextInt();
Employee employee = new Employee(employeeName, employeeAge);
c.add(employee);
}
// 2、创建迭代器遍历集合
Iterator it;
//3、遍历
while (it.hasnext) {
//4、集合中对象未知,向下转型
Employee e = it.next();
System.out.println(e.getName() + "---" + e.getAge());
}
}
}
输入样例:
在这里给出一组输入。例如:
zs
10
ls
20
ww
30
输出样例:
在这里给出相应的输出。例如:
zs---10
ls---20
ww---30
接下来是代码实现:
点击查看代码
import java.util.ArrayList;
import java.util.Collection;
import java.util.Iterator;
import java.util.Scanner;
// 定义员工类
class Employee {
private String name;
private int age;
public Employee() {
super();
}
public Employee(String name, int age) {
super();
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
}
// 主函数
public class Main {
public static void main(String[] args) {
// 1、创建有序集合对象
Collection<Employee> c = new ArrayList<>();
// 创建3个员工元素对象
Scanner sc = new Scanner(System.in);
for (int i = 0; i < 3; i++) {
String employeeName = sc.nextLine();
int employeeAge = Integer.parseInt(sc.nextLine());
Employee employee = new Employee(employeeName, employeeAge);
c.add(employee);
}
// 2、创建迭代器遍历集合
Iterator<Employee> it = c.iterator();
// 3、遍历
while (it.hasNext()) {
// 4、集合中对象未知,向下转型
Employee e = it.next();
System.out.println(e.getName()+"---"+e.getAge());
}
}
}
这道题非常简单我就不过多赘述了,接下来是新增的第一题,这道题目是关于电路设计,下面是题面:
智能家居是在当下家庭中越来越流行的一种配置方案,它通过物联网技术将家中的各种设备(如音视频设备、照明系统、窗帘控制、空调控制、安防系统、数字影院系统、影音服务器、影柜系统、网络家电等)连接到一起,提供家电控制、照明控制、电话远程控制、室内外遥控、防盗报警、环境监测、暖通控制、红外转发以及可编程定时控制等多种功能和手段。与普通家居相比,智能家居不仅具有传统的居住功能,兼备建筑、网络通信、信息家电、设备自动化,提供全方位的信息交互功能。请根据如下要去设计一个智能家居强电电路模拟系统。
1、控制设备模拟
本题模拟的控制设备包括:开关、分档调速器、连续调速器。
开关:包括0和1两种状态。
开关有两个引脚,任意一个引脚都可以是输入引脚,而另一个则是输出引脚。开关状态为0时,无论输入电位是多少,输出引脚电位为0。当开关状态为1时,输出引脚电位等于输入电位。
分档调速器
按档位调整,常见的有3档、4档、5档调速器,档位值从0档-2(3/4)档变化。本次迭代模拟4档调速器,每个档位的输出电位分别为0、0.3、0.6、0.9倍的输入电压。
连续调速器
没有固定档位,按位置比例得到档位参数,数值范围在[0.00-1.00]之间,含两位小数。输出电位为档位参数乘以输入电压。
所有调速器都有两个引脚,一个固定的输入(引脚编号为1)、一个输出引脚(引脚编号为2)。当输入电位为0时,输出引脚输出的电位固定为0,不受各类开关调节的影响。
所有控制设备的初始状态/档位为0。
控制设备的输入引脚编号为1,输出引脚编号为2。
2、受控设备模拟
本题模拟的受控设备包括:灯、风扇。两种设备都有两根引脚,通过两根引脚电压的电压差驱动设备工作。
灯有两种工作状态:亮、灭。在亮的状态下,有的灯会因引脚电位差的不同亮度会有区别。
风扇在接电后有两种工作状态:停止、转动。风扇的转速会因引脚的电位差的不同而有区别。
本次迭代模拟两种灯具。
白炽灯:
亮度在0~200lux(流明)之间。
电位差为0-9V时亮度为0,其他电位差按比例,电位差10V对应50ux,220V对应200lux,其他电位差与对应亮度值成正比。白炽灯超过220V。
日光灯:
亮度为180lux。
只有两种状态,电位差为0时,亮度为0,电位差不为0,亮度为180。
本次迭代模拟一种吊扇。
工作电压区间为80V-150V,对应转速区间为80-360转/分钟。80V对应转速为80转/分钟,150V对应转速为360转/分钟,超过150V转速为360转/分钟(本次迭代暂不考虑电压超标的异常情况)。其他电压值与转速成正比,输入输出电位差小于80V时转速为0。
输入信息:
1、设备信息
分别用设备标识符K、F、L、B、R、D分别表示开关、分档调速器、连续调速器、白炽灯、日光灯、吊扇。
设备标识用标识符+编号表示,如K1、F3、L2等。
引脚格式:设备标识-引脚编号,例如:K1-1标识编号为1的开关的输入引脚。
三种控制开关的输入引脚编号为1,输出引脚编号为2。
受控设备的两个引脚编号分别为1、2。
约束条件:
不同设备的编号可以相同。
同种设备的编号可以不连续。
设备信息不单独输入,包含在连接信息中。
2、连接信息
一条连接信息占一行,用[]表示一组连接在一起的设备引脚,引脚与引脚之间用英文空格" "分隔。
格式:"["+引脚号+" "+...+" "+引脚号+"]"
例如:[K1-1 K3-2 D5-1]表示K1的输入引脚,K3的输出引脚,D5的1号引脚连接在一起。
约束条件:
本次迭代不考虑两个输出引脚短接的情况
考虑调速器输出串联到其他控制设备(开关)的情况
不考虑调速器串联到其他调速器的情况。
不考虑各类控制设备的并联接入或反馈接入。例如,K1的输出接到L2的输入,L2的输出再接其他设备属于串联接线。K1的输出接到L2的输出,同时K1的输入接到L2的输入,这种情况属于并联。K1的输出接到L2的输入,K1的输入接到L2的输出,属于反馈接线。
3、控制设备调节信息
开关调节信息格式:
+设备标识K+设备编号,例如:#K2,代表切换K2开关的状态。
分档调速器的调节信息格式:
+设备标识F+设备编号+"+" 代表加一档,例如:#F3+,代表F3输出加一档。
+设备标识F+设备编号+"-" 代表减一档,例如:#F1-,代表F1输出减一档。
连续调速器的调节信息格式:
+设备标识L+设备编号+":" +数值 代表将连续调速器的档位设置到对应数值,例如:#L3:0.6,代表L3输出档位参数0.6。
4、电源接地标识:VCC,电压220V,GND,电压0V。没有接线的引脚默认接地,电压为0V。
输入信息以end为结束标志,忽略end之后的输入信息。
输出信息:
按开关、分档调速器、连续调速器、白炽灯、日光灯、吊扇的顺序依次输出所有设备的状态或参数。每个设备一行。同类设备按编号顺序从小到大输出。
输出格式:@设备标识+设备编号+":" +设备参数值(控制开关的档位或状态、灯的亮度、风扇的转速,只输出值,不输出单位)
连续调速器的档位信息保留两位小数,即使小数为0,依然显示两位小数.00。
开关状态为0(打开)时显示turned on,状态为1(合上)时显示closed
如:
@K1:turned on
@B1:190
@L1:0.60
本题不考虑输入电压或电压差超过220V的情况。
本题只考虑串联的形式,所以所有测试用例的所有连接信息都只包含两个引脚
本题电路中除了开关可能出现多个,其他电路设备均只出现一次。
电源VCC一定是第一个连接的第一项,接地GND一定是最后一个连接的后一项。
家居电路模拟系列所有题目的默认规则:
1、当计算电压值等数值的过程中,最终结果出现小数时,用截尾规则去掉小数部分,只保留整数部分。为避免精度的误差,所有有可能出现小数的数值用double类型保存并计算,不要作下转型数据类型转换,例如电压、转速、亮度等,只有在最后输出时再把计算结果按截尾规则,舍弃尾数,保留整数输出。
2、所有连接信息按电路从电源到接地的顺序依次输入,不会出现错位的情况。
3、连接信息如果只包含两个引脚,靠电源端的引脚在前,靠接地端的在后。
4、对于调速器,其输入端只会直连VCC,不会接其他设备。整个电路中最多只有一个调速器,且连接在电源上。
家居电路模拟系列1-4题目后续迭代设计:
1、电路结构变化:
迭代1:只有一条线路,所有元件串联
迭代2:线路中包含一个并联电路
迭代3:线路中包含多个串联起来的并联电路
迭代4:并联电路之间可能出现包含关系
电路结构变化示意图见图1。
2、输入信息的变化
串联线路信息:用于记录一段串联电路的元件与连接信息。
例如: #T1:[IN K1-1] [K1-2 D2-1] [D2-2 OUT]
#T1:[IN K1-1] [K1-2 M1-IN][M1-OUT D2-1] [D2-2 GND]
并联线路信息:用于记录一段并联电路所包含的所有串联电路信息。
例如:#M1:[T1 T2 T3]
以上格式仅做参考,格式细节可能会调整,以具体发布的为准。
3、计算方式的变化
迭代1只包含1个受控元件,不用计算电流,之后的电路计算要包含电流、电阻等电路参数。
4、电路元件的变化
每次迭代会增加1-2个新的电路元件。
image.png
图1:电路结构示意图
设计建议:
1、电路设备类:描述所有电路设备的公共特征。
2、受控设备类、控制设备类:对应受控、控制设备
3、串联电路类:一条由多个电路设备构成的串联电路,也看成是一个独立的电路设备
其他类以及类的属性、方法自行设计。
image.png
图2:建议设计类图
输入样例1:
在这里给出一组输入。例如:
[VCC K1-1]
[K1-2 D2-1]
[D2-2 GND]
K1
end
输出样例1:
在这里给出相应的输出。例如:
@K1:closed
@D2:360
输入样例2:
在这里给出一组输入。例如:
[VCC K1-1]
[K1-2 D2-1]
[D2-2 GND]
K1
K1
end
输出样例2:
在这里给出相应的输出。例如:
@K1:turned on
@D2:0
输入样例3:
在这里给出一组输入。例如:
[VCC F1-1]
[F1-2 D2-1]
[D2-2 GND]
F1+
end
输出样例3:
在这里给出相应的输出。例如:
@F1:1
@D2:0
输入样例4:
在这里给出一组输入。例如:
[VCC F1-1]
[F1-2 D2-1]
[D2-2 GND]
F1+
F1+
end
输出样例4:
在这里给出相应的输出。例如:
@F1:2
@D2:288
输入样例5:
在这里给出一组输入。例如:
[VCC F1-1]
[F1-2 D2-1]
[D2-2 GND]
F1+
F1+
F1+
end
输出样例5:
在这里给出相应的输出。例如:
@F1:3
@D2:360
输入样例6:
在这里给出一组输入。例如:
[VCC L1-1]
[L1-2 D2-1]
[D2-2 GND]
L1:1.00
end
输出样例6:
在这里给出相应的输出。例如:
@L1:1.00
@D2:360
输入样例7:
在这里给出一组输入。例如:
[VCC L1-1]
[L1-2 D2-1]
[D2-2 GND]
L1:0.68
end
输出样例7:
在这里给出相应的输出。例如:
@L1:0.68
@D2:358
输入样例8:
在这里给出一组输入。例如:
[VCC L1-1]
[L1-2 B2-1]
[B2-2 GND]
L1:0.68
end
输出样例8:
在这里给出相应的输出。例如:
@L1:0.68
@B2:149
输入样例9:
在这里给出一组输入。例如:
[VCC L1-1]
[L1-2 B2-1]
[B2-2 GND]
L1:1.00
end
输出样例9:
在这里给出相应的输出。例如:
@L1:1.00
@B2:200
输入样例10:
在这里给出一组输入。例如:
[VCC L1-1]
[L1-2 R2-1]
[R2-2 GND]
end
输出样例10:
在这里给出相应的输出。例如:
@L1:1.00
@R2:180
关于这道题目其实我想说,这个引脚之间的电压差是非常重要的一点,因为受控设备的灯和风扇都随着引脚间电压差的变化而发生改变,同时这道题目由于不需要考虑短接以及其他的并联等情况,所以一开始看上去还是比较简单的。
下面是我的代码:
点击查看代码
import java.util.*;
abstract class Device {
protected String deviceType;
protected int inputPin;
protected int outputPin;
protected double inputVoltage = 0;
protected String currentState;
public Device(String deviceType, int inputPin, int outputPin) {
this.deviceType = deviceType;
this.inputPin = inputPin;
this.outputPin = outputPin;
this.currentState = "0";
}
public abstract void setState(String state);
public abstract double calculateOutput();
public String getDeviceType() {
return deviceType;
}
public double getInputVoltage() {
return inputVoltage;
}
public void setInputVoltage(double inputVoltage) {
this.inputVoltage = inputVoltage;
}
public String getCurrentState() {
return currentState;
}
}
class Switch extends Device {
public Switch(int inputPin, int outputPin) {
super("Switch", inputPin, outputPin);
}
@Override
public void setState(String state) {
this.currentState = state;
}
@Override
public double calculateOutput() {
return currentState.equals("1") ? inputVoltage : 0;
}
@Override
public String getCurrentState() {
return currentState.equals("1") ? "closed" : "turned on";
}
}
class StepRegulator extends Device {
private static final double[] OUTPUT_MULTIPLIERS = {0, 0.3, 0.6, 0.9};
private int level;
public StepRegulator(int inputPin, int outputPin) {
super("StepRegulator", inputPin, outputPin);
this.level = 0;
}
@Override
public void setState(String state) {
if (state.equals("+")) {
level = Math.min(level + 1, OUTPUT_MULTIPLIERS.length - 1);
} else if (state.equals("-")) {
level = Math.max(level - 1, 0);
}
this.currentState = Integer.toString(level);
}
@Override
public double calculateOutput() {
return inputVoltage * OUTPUT_MULTIPLIERS[level];
}
}
class ContinuousRegulator extends Device {
private double position;
public ContinuousRegulator(int inputPin, int outputPin) {
super("ContinuousRegulator", inputPin, outputPin);
this.position = 0.0;
}
@Override
public void setState(String state) {
this.position = Double.parseDouble(state);
this.currentState = String.format("%.2f", position);
}
@Override
public double calculateOutput() {
return inputVoltage * position;
}
}
abstract class ControlledDevice extends Device {
public ControlledDevice(String deviceType, int inputPin, int outputPin) {
super(deviceType, inputPin, outputPin);
}
}
class IncandescentLight extends ControlledDevice {
public IncandescentLight(int inputPin, int outputPin) {
super("IncandescentLight", inputPin, outputPin);
}
@Override
public void setState(String state) {
// Not applicable for this type
}
@Override
public double calculateOutput() {
double voltageDifference = inputVoltage;
if (voltageDifference <= 9) {
return 0;
}
return Math.min(200, (voltageDifference / 220) * 200);
}
}
class FluorescentLight extends ControlledDevice {
public FluorescentLight(int inputPin, int outputPin) {
super("FluorescentLight", inputPin, outputPin);
}
@Override
public void setState(String state) {
// Not applicable for this type
}
@Override
public double calculateOutput() {
double voltageDifference = inputVoltage;
return voltageDifference > 0 ? 180 : 0;
}
}
class Fan extends ControlledDevice {
public Fan(int inputPin, int outputPin) {
super("Fan", inputPin, outputPin);
}
@Override
public void setState(String state) {
// Not applicable for this type
}
@Override
public double calculateOutput() {
double voltageDifference = inputVoltage;
if (voltageDifference < 80) {
return 0;
} else if (voltageDifference > 150) {
return 360;
} else {
return (voltageDifference - 80) / 70 * 280 + 80;
}
}
}
public class Main {
private static Map<String, Device> devices = new HashMap<>();
private static Map<String, List<String>> connections = new HashMap<>();
private static Map<String, Double> voltages = new HashMap<>();
public static void main(String[] args) {
Scanner scanner = new Scanner(System.in);
while (scanner.hasNext()) {
String line = scanner.nextLine().trim();
if (line.equals("end")) {
break;
}
if (line.startsWith("[")) {
parseConnection(line);
} else if (line.startsWith("#")) {
parseControl(line);
}
}
updateVoltages();
printDeviceStates();
}
private static void parseConnection(String line) {
line = line.substring(1, line.length() - 1).trim();
String[] parts = line.split(" ");
for (String part : parts) {
connections.putIfAbsent(part, new ArrayList<>());
for (String p : parts) {
if (!p.equals(part)) {
connections.get(part).add(p);
}
}
}
for (String part : parts) {
if (part.startsWith("VCC")) {
voltages.put(part, 220.0);
} else if (part.startsWith("GND")) {
voltages.put(part, 0.0);
} else {
String deviceType = part.substring(0, 1);
int deviceId = Integer.parseInt(part.substring(1, part.indexOf('-')));
int pin = Integer.parseInt(part.substring(part.indexOf('-') + 1));
Device device;
switch (deviceType) {
case "K":
device = new Switch(1, 2);
break;
case "F":
device = new StepRegulator(1, 2);
break;
case "L":
device = new ContinuousRegulator(1, 2);
break;
case "B":
device = new IncandescentLight(1, 2);
break;
case "R":
device = new FluorescentLight(1, 2);
break;
case "D":
device = new Fan(1, 2);
break;
default:
continue;
}
devices.put(deviceType + deviceId, device);
}
}
}
private static void parseControl(String line) {
String deviceId = line.substring(1, 3);
Device device = devices.get(deviceId);
if (device instanceof Switch) {
((Switch) device).setState(device.getCurrentState().equals("1") ? "0" : "1");
} else if (device instanceof StepRegulator) {
((StepRegulator) device).setState(line.charAt(3) == '+' ? "+" : "-");
} else if (device instanceof ContinuousRegulator) {
((ContinuousRegulator) device).setState(line.substring(4));
}
}
private static void updateVoltages() {
for (String key : connections.keySet()) {
if (voltages.containsKey(key)) {
updateVoltage(key, voltages.get(key));
}
}
}
private static void updateVoltage(String pin, double voltage) {
voltages.put(pin, voltage);
for (String connectedPin : connections.get(pin)) {
if (!voltages.containsKey(connectedPin)) {
voltages.put(connectedPin, voltage);
String deviceId = connectedPin.substring(0, connectedPin.indexOf('-'));
Device device = devices.get(deviceId);
if (device != null) {
if (connectedPin.endsWith("1")) {
device.setInputVoltage(voltage);
}
}
updateVoltage(connectedPin, voltage);
}
}
}
private static void printDeviceStates() {
List<String> deviceTypes = Arrays.asList("K", "F", "L", "B", "R", "D");
for (String type : deviceTypes) {
devices.entrySet().stream()
.filter(e -> e.getKey().startsWith(type))
.sorted(Comparator.comparingInt(e -> Integer.parseInt(e.getKey().substring(1))))
.forEach(e -> {
Device device = e.getValue();
String state = device instanceof ContinuousRegulator ?
String.format("%.2f", Double.parseDouble(device.getCurrentState())) :
device instanceof Switch ? device.getCurrentState() :
String.valueOf((int) device.calculateOutput());
System.out.println("@" + e.getKey() + ":" + state);
});
}
}
}
下面是我的类图:
在这个电路设计上我吸取了上一次设计的教训,避开了我的缺点,所以我这次设计了多个类使它们各司其职,从而较好的完成单一职责,具体的类间关系以及他们各自的方法之类的我就不过多赘述了,可以从我的类图中看出。关于这次的题目,我认为这道题目的关键点在于处理状态管理和设备的初始状态和状态切换逻辑;以及电压传播,因为它根据连接关系传播电压,从而影响设备状态;还有指令处理,包括解析和执行控制设备的调节指令;最后是输出生成,要按要求输出所有设备的当前状态或参数。虽然话是这么说,但是我没有办法把这些都给实现,所以这次相对简单的大作业我也没有达到要求,但是我也在努力寻找解题的关键点,希望能慢慢得高分。接下来是第六次大作业。
第六次PTA作业:
这次的作业就只有一道家具强电电路问题了,也是延续上一次的题目,话不多说直接上题目要求:
智能家居是在当下家庭中越来越流行的一种配置方案,它通过物联网技术将家中的各种设备(如音视频设备、照明系统、窗帘控制、空调控制、安防系统、数字影院系统、影音服务器、影柜系统、网络家电等)连接到一起,提供家电控制、照明控制、电话远程控制、室内外遥控、防盗报警、环境监测、暖通控制、红外转发以及可编程定时控制等多种功能和手段。与普通家居相比,智能家居不仅具有传统的居住功能,兼备建筑、网络通信、信息家电、设备自动化,提供全方位的信息交互功能。请根据如下要去设计一个智能家居强电电路模拟系统。以下题目介绍中加粗的部分为本次迭代在“家居强电电路模拟程序-1”的基础上增加的功能要求。
1、控制设备
本题模拟的控制设备包括:开关、分档调速器、连续调速器。
开关:包括0和1两种状态。
开关有两个引脚,任意一个引脚都可以是输入引脚,而另一个则是输出引脚。开关状态为0时,无论输入电位是多少,输出引脚电位为0。当开关状态为1时,输出引脚电位等于输入电位。
分档调速器
按档位调整,常见的有3档、4档、5档调速器,档位值从0档-2(3/4)档变化。本次迭代模拟4档调速器,每个档位的输出电位分别为0、0.3、0.6、0.9倍的输入电压。
连续调速器
没有固定档位,按位置比例得到档位参数,数值范围在[0.00-1.00]之间,含两位小数。输出电位为档位参数乘以输入电压。
所有调速器都有两个引脚,一个固定的输入(引脚编号为1)、一个输出引脚(引脚编号为2)。当输入电位为0时,输出引脚输出的电位固定为0,不受各类开关调节的影响。
所有控制设备的初始状态/档位为0。
控制设备的输入引脚编号为1,输出引脚编号为2。
所有开关的电阻为 0。
2、受控设备
本题模拟的受控设备包括:灯、风扇。两种设备都有两根引脚,通过两根引脚电压的电压差驱动设备工作。
灯有两种工作状态:亮、灭。在亮的状态下,有的灯会因引脚电位差的不同亮度会有区别。
风扇在接电后有两种工作状态:停止、转动。风扇的转速会因引脚间电位差的不同而有区别。
本次迭代模拟两种灯具。
白炽灯:
亮度在0~200lux(流明)之间。
电位差为0-9V时亮度为0,其他电位差按比例,电位差10V对应50ux,220V对应200lux,其他电位差与对应亮度值成正比。白炽灯超过220V。
日光灯:
亮度为180lux。
只有两种状态,电位差为0时,亮度为0,电位差不为0,亮度为180。
本次迭代模拟一种吊扇。
工作电压区间为80V-150V,对应转速区间为80-360转/分钟。80V对应转速为80转/分钟,150V对应转速为360转/分钟,超过150V转速为360转/分钟(本次迭代暂不考虑电压超标的异常情况)。其他电压值与转速成正比,输入输出电位差小于80V时转速为0。
本次迭代模拟一种落地扇。
工作电压区间为 [80V,150V],对应转速区间为 80-360 转/分钟。电压在[80,100)V 区间对应转速为 80 转/分 钟,[100-120)V 区间对应转速为 160 转/分钟,[120-140)V 区间对应转速为 260 转/分钟,超过 140V 转速 为 360 转/分钟(本次迭代暂不考虑电压超标的异常情况)输入信息:
本次迭代考虑电阻:白炽灯的电阻为 10,日光灯的电阻为 5,吊扇的电阻为 20,落 地扇的电阻为 20
3、输入信息
1)输入设备信息
分别用设备标识符K、F、L、B、R、D、A分别表示开关、分档调速器、连续调速器、白炽灯、日光灯、吊扇、落地扇。
设备标识用标识符+编号表示,如K1、F3、L2等。
引脚格式:设备标识-引脚编号,例如:K1-1标识编号为1的开关的输入引脚。
三种控制开关的输入引脚编号为1,输出引脚编号为2。
受控设备的两个引脚编号分别为1、2。
约束条件:
不同设备的编号可以相同。
同种设备的编号可以不连续。
设备信息不单独输入,包含在连接信息中。
2)输入连接信息
一条连接信息占一行,用[]表示一组连接在一起的设备引脚,引脚与引脚之间用英文空格" "分隔。
格式:"["+引脚号+" "+...+" "+引脚号+"]"
例如:[K1-1 K3-2 D5-1]表示K1的输入引脚,K3的输出引脚,D5的1号引脚连接在一起。
约束条件:
不考虑调速器串联到其他调速器的情况。
不考虑调速器串联到其他调速器的情况。
考虑各类设备的并联接入。例如,K1 的输出接到 L2 的输入,L2 的输出再接其他设备属于串联接线。K1 的输出接到 L2 的输出,同时 K1 的输入接到 L2 的输入,这种情况属于并联。
本次迭代的连接信息不单独输入,包含在线路信息中。
3)输入控制设备调节信息
开关调节信息格式:
+设备标识K+设备编号,例如:#K2,代表切换K2开关的状态。
分档调速器的调节信息格式:
+设备标识F+设备编号+"+" 代表加一档,例如:#F3+,代表F3输出加一档。
+设备标识F+设备编号+"-" 代表减一档,例如:#F1-,代表F1输出减一档。
连续调速器的调节信息格式:
+设备标识L+设备编号+":" +数值 代表将连续调速器的档位设置到对应数值,例如:#L3:0.6,代表L3输出档位参数0.6。
4)电源接地标识:
VCC,电压220V,GND,电压0V。没有接线的引脚默认接地,电压为0V。
5)输入串联电路信息
一条串联电路占一行,串联电路由按从靠电源端到接地端顺序依次输入的 n 个连接 信息组成,连接信息之间用英文空格" "分隔。
串联电路信息格式:
"#T"+电路编号+":"+连接信息+" "+连接信息+...+" "+连接信息
例如:#T1:[IN K1-1] [K1-2 D2-1] [D2-2 OUT] 一个串联电路的第一个引脚是 IN,代表起始端,靠电源。最后一个引脚是 OUT,代表结尾端, 靠接地。
约束条件:
不同的串联电路信息编号不同。
输入的最后一条电路信息必定是总电路信息,总电路信息的起始引脚是 VCC,结束引脚是 GND。
连接信息中的引脚可能是一条串联或并联电路的 IN 或者 OUT。例如:
T1:[IN K1-1] [K1-2 T2-IN] [T2-OUT OUT]
T1:[IN K1-1] [K1-2 T2-IN] [T2-OUT M2-IN] [M2-OUT OUT]
6)输入并联电路信息
一条并联电路占一行,并联电路由其包含的几条串联电路组成,串联电路标识之间用英文空格" "分隔。
格式:
"#M"+电路编号+":"+”[”+串联电路信息+" "+....+" "+串联电路信息+”]”
例如:#M1:[T1 T2 T3]
该例声明了一个并联电路,由 T1、T2、T3 三条串联电路并联而成,三条串联电路的 IN 短 接在一起构成 M1 的 IN,三条串联电路的 OUT 短接在一起构成 M1 的 OUT。
约束条件:
本次迭代不考虑并联电路中包含并联电路的情况,也不考虑多个并联电路串联的情况。
本题不考虑输入电压或电压差超过220V的情况。
输入信息以end为结束标志,忽略end之后的输入信息。
本题中的并联信息所包含的串联电路的信息都在并联信息之前输入,不考虑乱序输入的情况。
电路中的短路如果不会在电路中产生无穷大的电流烧坏电路,都是合理情况,在本题测试点的考虑范围之内。
本题不考虑一条串联电路中包含其他串联电路的情况。例如:
T3:[VCC K1-1] [K1-2 T2-IN] [T2-OUT K2-1] [K2-2 T1-IN] [T1-OUT GND]
本例中T1\T2两条串联电路实际是T3的一个部分,本题不考虑这种类型的输入,而是当将T1\T2的所有连接信息直接包含在T3中定义。
下次迭代中需要考虑这种类型的输入。
4、输出信息:
按开关、分档调速器、连续调速器、白炽灯、日光灯、吊扇、落地扇的顺序依次输出所有设备的状态或参数。每个设备一行。同类设备按编号顺序从小到大输出。
输出格式:@设备标识+设备编号+":" +设备参数值(控制开关的档位或状态、灯的亮度、风扇的转速,只输出值,不输出单位)
连续调速器的档位信息保留两位小数,即使小数为0,依然显示两位小数.00。
开关状态为0(打开)时显示turned on,状态为1(合上)时显示closed
如:
@K1:turned on
@B1:190
@L1:0.60
5、家居电路模拟系列所有题目的默认规则:
1)当计算电压值等数值的过程中,最终结果出现小数时,用截尾规则去掉小数部分,只保留整数部分。为避免精度的误差,所有有可能出现小数的数值用double类型保存并计算,不要作下转型数据类型转换,例如电压、转速、亮度等,只有在最后输出时再把计算结果按截尾规则,舍弃尾数,保留整数输出。
2)所有连接信息按电路从电源到接地的顺序依次输入,不会出现错位的情况。电源VCC一定是第一个连接的第一项,接地GND一定是最后一个连接的后一项。
3)连接信息如果只包含两个引脚,靠电源端的引脚在前,靠接地端的在后。
4)调速器的输入端只会直连VCC,不会接其他设备。整个电路最多只有连接在电源上的一个调速器,且不包含在并联单路中。
6、家居电路模拟系列1-4题目后续迭代设计:
1)电路结构变化:
迭代1:只有一条线路,所有元件串联
迭代2:线路中包含一个并联电路
迭代3:线路中包含多个串联起来的并联电路
迭代4:并联电路之间可能出现包含关系
电路结构变化示意图见图1。
2)计算方式的变化
迭代1只包含1个受控元件,不用计算电流,之后的电路计算要包含电流、电阻等电路参数。
3)电路元件的变化
每次迭代会增加1-2个新的电路元件。
image.png
图1:电路结构示意图
设计建议:
1、电路设备类:描述所有电路设备的公共特征。
2、受控设备类、控制设备类:对应受控、控制设备
3、串联电路类:一条由多个电路设备构成的串联电路,也看成是一个独立的电路设备
4、并联电路类:继承电路设备类,也看成是一个独立的电路设备
其他类以及类的属性、方法自行设计。
image.png
图2:建议设计类图
输入格式:
请在这里写输入格式。例如:输入在一行中给出2个绝对值不超过1000的整数A和B。
输出格式:
请在这里描述输出格式。例如:对每一组输入,在一行中输出A+B的值。
输入样例1:
在这里给出一组输入。例如:
T1:[IN K1-1] [K1-2 D2-1] [D2-2 OUT]
T2:[IN K2-1] [K2-2 D1-1] [D1-2 OUT]
M1:[T1 T2]
T3:[VCC L1-1] [L1-2 M1-IN] [M1-OUT D3-1] [D3-2 GND]
K1
end
输出样例1:
在这里给出相应的输出。例如:
@K1:closed
@K2:turned on
@L1:0.00
@D1:0
@D2:0
@D3:0
输入样例2:
在这里给出一组输入。例如:
T1:[IN K1-1] [K1-2 D2-1] [D2-2 OUT]
T2:[IN K2-1] [K2-2 D1-1] [D1-2 OUT]
M1:[T1 T2]
T3:[VCC L1-1] [L1-2 M1-IN] [M1-OUT D3-1] [D3-2 GND]
K1
L1:1.00
end
输出样例2:
在这里给出相应的输出。例如:
@K1:closed
@K2:turned on
@L1:1.00
@D1:0
@D2:200
@D3:200
输入样例3:
在这里给出一组输入。例如:
T1:[IN K1-1] [K1-2 D2-1] [D2-2 OUT]
T2:[IN K2-1] [K2-2 D1-1] [D1-2 OUT]
M1:[T1 T2]
T3:[VCC L1-1] [L1-2 M1-IN] [M1-OUT D3-1] [D3-2 GND]
K1
K2
L1:1.00
end
输出样例3:
在这里给出相应的输出。例如:
@K1:closed
@K2:closed
@L1:1.00
@D1:0
@D2:0
@D3:346
这一次给出的测试点较少,输入输出样例也少。我认为这道题目还是比较棘手的,接下来是我修改后的新代码,但是编译器没有按预期生成二进制文件,所以我这道题目并没有通过测试点。
点击查看代码
import java.util.ArrayList;
import java.util.Comparator;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.*;
class Circuit {
private Map<String, Device> devices = new HashMap<>();
private List<List<String>> connections = new ArrayList<>();
public void addDevice(Device device) {
devices.put(device.getId(), device);
}
public Device getDevice(String id) {
return devices.get(id);
}
public void addConnection(List<String> connection) {
connections.add(connection);
}
public void updateVoltage() {
for (List<String> connection : connections) {
double voltage = 220;
for (String pin : connection) {
if (pin.contains("-")) { // 检查连接信息是否包含分隔符
String[] parts = pin.split("-");
String deviceId = parts[0];
int pinNumber = Integer.parseInt(parts[1]);
Device device = devices.get(deviceId);
if (device != null) {
if (pinNumber == 1) {
device.setVoltage(voltage);
} else {
voltage = device.getVoltage();
}
}
} else {
System.out.println("Invalid connection format: " + pin);
}
}
}
}
public void printStatus() {
devices.values().stream()
.sorted(Comparator.comparing(Device::getId))
.forEach(device -> System.out.println("@" + device.getId() + ":" + device.getStatus()));
}
}
abstract class AbstractDevice implements Device {
protected String id;
protected double voltage;
public AbstractDevice(String id) {
this.id = id;
this.voltage = 0.0;
}
public String getId() {
return id;
}
public double getVoltage() {
return voltage;
}
public void setVoltage(double voltage) {
this.voltage = voltage;
}
public abstract String getStatus();
}
class CeilingFan extends Fan {
public CeilingFan(String id) {
super(id);
}
@Override
public double getVoltage() {
if (voltage < 80) return 0;
return Math.min(360, ((voltage - 80) / (150 - 80)) * (360 - 80) + 80);
}
@Override
public String getStatus() {
return String.valueOf((int) getVoltage());
}
}
class ContinuousController extends AbstractDevice {
private double level;
public ContinuousController(String id) {
super(id);
this.level = 0.0;
}
public void setLevel(double level) {
if (level >= 0.0 && level <= 1.0) {
this.level = level;
}
}
@Override
public double getVoltage() {
return voltage * level;
}
@Override
public String getStatus() {
return String.format("%.2f", level);
}
}
interface Device {
String getId();
double getVoltage();
void setVoltage(double voltage);
String getStatus();
}
abstract class Fan extends AbstractDevice {
public Fan(String id) {
super(id);
}
protected double calculateSpeed(double voltage) {
return 0.0;
}
}
class FloorFan extends Fan {
public FloorFan(String id) {
super(id);
}
@Override
public double getVoltage() {
if (voltage < 80) return 0;
else if (voltage < 100) return 80;
else if (voltage < 120) return 160;
else if (voltage < 140) return 260;
else return 360;
}
@Override
public String getStatus() {
return String.valueOf((int) getVoltage());
}
}
class FluorescentLight extends Light {
public FluorescentLight(String id) {
super(id);
}
@Override
public double getVoltage() {
return voltage == 0 ? 0 : 180;
}
@Override
public String getStatus() {
return String.valueOf((int) getVoltage());
}
}
class IncandescentLight extends Light {
public IncandescentLight(String id) {
super(id);
}
@Override
public double getVoltage() {
if (voltage < 10) return 0;
return Math.min(200, (voltage / 220) * 200);
}
@Override
public String getStatus() {
return String.valueOf((int) getVoltage());
}
}
abstract class Light extends AbstractDevice {
public Light(String id) {
super(id);
}
protected double calculateBrightness(double voltage) {
return 0.0;
}
}
class SmartHomeSimulation {
public static void main(String[] args) {
Scanner scanner = new Scanner(System.in);
Circuit circuit = new Circuit();
while (scanner.hasNextLine()) {
String line = scanner.nextLine().trim();
if (line.equals("end")) break;
if (line.startsWith("#")) {
// Process control commands
if (line.contains(":")) {
String[] parts = line.split(":");
String command = parts[0];
String info = parts[1];
if (command.startsWith("#T")) {
// Handle series circuit
List<String> connection = Arrays.asList(info.replace("[", "").replace("]", "").split(" "));
circuit.addConnection(connection);
} else if (command.startsWith("#M")) {
// Handle parallel circuit
String[] seriesCircuits = info.replace("[", "").replace("]", "").split(" ");
for (String seriesCircuit : seriesCircuits) {
List<String> connection = Arrays.asList(seriesCircuit.split(" "));
circuit.addConnection(connection);
}
}
} else {
// Handle device adjustment
char type = line.charAt(1);
String id = line.substring(2);
if (type == 'K') {
Switch sw = (Switch) circuit.getDevice(id);
if (sw != null) {
sw.toggle();
}
} else if (type == 'F') {
StepController sc = (StepController) circuit.getDevice(id);
if (line.endsWith("+")) {
sc.increaseLevel();
} else if (line.endsWith("-")) {
sc.decreaseLevel();
}
} else if (type == 'L') {
ContinuousController cc = (ContinuousController) circuit.getDevice(id);
double level = Double.parseDouble(line.split(":")[1]);
cc.setLevel(level);
}
}
} else if (line.startsWith("[")) {
// Process connection information
List<String> connection = Arrays.asList(line.replace("[", "").replace("]", "").split(" "));
circuit.addConnection(connection);
} else {
// Add device
String[] parts = line.split(" ");
for (String part : parts) {
char type = part.charAt(0);
String id = part.substring(1);
switch (type) {
case 'K':
circuit.addDevice(new Switch(id));
break;
case 'F':
circuit.addDevice(new StepController(id));
break;
case 'L':
circuit.addDevice(new ContinuousController(id));
break;
case 'B':
circuit.addDevice(new IncandescentLight(id));
break;
case 'R':
circuit.addDevice(new FluorescentLight(id));
break;
case 'D':
circuit.addDevice(new CeilingFan(id));
break;
case 'A':
circuit.addDevice(new FloorFan(id));
break;
}
}
}
}
circuit.updateVoltage();
circuit.printStatus();
}
}
class StepController extends AbstractDevice {
private int level;
private double[] levels = {0.0, 0.3, 0.6, 0.9};
public StepController(String id) {
super(id);
this.level = 0;
}
public void increaseLevel() {
if (level < levels.length - 1) level++;
}
public void decreaseLevel() {
if (level > 0) level--;
}
@Override
public double getVoltage() {
return voltage * levels[level];
}
@Override
public String getStatus() {
return String.valueOf(level);
}
}
class Switch extends AbstractDevice {
private boolean isOn;
public Switch(String id) {
super(id);
this.isOn = false;
}
public void toggle() {
this.isOn = !this.isOn;
}
@Override
public double getVoltage() {
return isOn ? voltage : 0;
}
@Override
public String getStatus() {
return isOn ? "turned on" : "closed";
}
}
关于这道题目呢,我认为还是在于处理状态管理和设备的初始状态和状态切换逻辑的问题上需要下心思去琢磨。
三、踩坑心得
我们在写代码的时候经常可能会出现数组越界的问题,就像是我的Circuit类中的updateVoltage方法中,就因为疏忽而出现了这个问题,这是我之前的代码:
点击查看代码
public void updateVoltage() {
for (List<String> connection : connections) {
double voltage = 220;
for (String pin : connection) {
String deviceId = pin.split("-")[0];
int pinNumber = Integer.parseInt(pin.split("-")[1]); // 这里出现越界异常
Device device = devices.get(deviceId);
if (device != null) {
if (pinNumber == 1) {
device.setVoltage(voltage);
} else {
voltage = device.getVoltage();
}
}
}
}
}
点击查看代码
public void updateVoltage() {
for (List<String> connection : connections) {
double voltage = 220;
for (String pin : connection) {
if (pin.contains("-")) { // 检查连接信息是否包含分隔符
String[] parts = pin.split("-");
String deviceId = parts[0];
int pinNumber = Integer.parseInt(parts[1]);
Device device = devices.get(deviceId);
if (device != null) {
if (pinNumber == 1) {
device.setVoltage(voltage);
} else {
voltage = device.getVoltage();
}
}
} else {
System.out.println("Invalid connection format: " + pin);
}
}
}
}
四、改进建议:
我认为呢,改进的不应该是题目,而是自己,自己的思维要进行转变和更新。
五、总结:
现在作业也已经做到了第六次,还剩最后几次就结束了,现在这个新的电路设计是可以很好的帮助我这种之前并没有很好的完成前面一次设计的同学的,通过这些程序的编写啊,课程的设计啊,以及配套的教学,我认为我们是可以学习到很多东西的。类似于在程序中使用类和对象的概念,将设备、电路等抽象为对象,并实现它们之间的交互和功能。又或者是使用合适的数据结构来存储设备信息和连接信息。还有学习如何处理可能出现的异常情况,例如数组越界异常、设备未找到等,保证程序的稳定性和健壮性,以及如何进行文件读写操作。这些都是我们以后可能会用到的技能。最后我想说还是加油吧,最起码也要跟上大家的步伐!共同进步!!!
标签:输出,题目,String,引脚,南昌,blog,23201823,public,输入 From: https://www.cnblogs.com/ruing1231/p/18237586