首页 > 其他分享 >PTA题目集4~6的总结

PTA题目集4~6的总结

时间:2024-06-09 14:21:41浏览次数:23  
标签:总结 输出 题目 引脚 PTA 电路 输入 设备

  1. 前言
  • [ 1 ] 知识点:本次PTA主要是新增的知识点是类的继承和多态,第4次题目集是针对类的继承,第5,6次题目集是针对类的多态。
  • [ 2 ] 题量:第4次和第5次题目集都是三题,第6次题目集是一题,题量不是很大
  • [ 3 ] 难度:第4次题目集还是答题判题程序,难度还行,就是增加了选择题和填空题,我感觉难点就是多选和多填的情况。第5,6次题目集是家居强电电路模拟程序。这个程序的难点就在于会有串并联电路连接和计算各个部分的分压。
  1. 设计与分析
  • [ 1 ] 答题判题程序-4
    题目要求:设计实现答题程序,模拟一个小型的测试,要求输入题目信息、试卷信息、答题信息、学生信息、删除题目信息,根据输入题目信息中的标准答案判断答题的结果。
点击查看题目具体要求
输入格式: 

程序输入信息分五种,信息可能会打乱顺序混合输入。

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=~5~false
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=~5~false
党十八大报告提出要加强()建设。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=~5~false
古琴在古代被称为:~瑶琴~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=~5~false
古琴在古代被称为:~古筝~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=~5~false
古琴在古代被称为:~瑶琴或七弦琴~true
20201103 Tom: 0 10~10
1+1=~2~true
古琴在古代被称为:~古筝~false
20201103 Tom: 8 0~8
1+1=~2~true
古琴在古代被称为:~瑟~false
20201104 Jack: 5 0~5
1+1=~5~false
古琴在古代被称为:~七弦琴~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=~5~false
党十八大报告提出要加强()建设。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=~5~false
古琴在古代被称为:~瑶琴~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=~5~false
古琴在古代被称为:~古筝~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=~5~false
古琴在古代被称为:~瑶琴或七弦琴~true
20201103 Tom: 0 10~10
1+1=~2~true
古琴在古代被称为:~古筝~false
20201103 Tom: 8 0~8
1+1=~2~true
古琴在古代被称为:~瑟~false
20201104 Jack: 5 0~5
1+1=~5~false
古琴在古代被称为:~七弦琴~partially correct
20201104 Jack: 0 10~10
类图分析:

class ZQuestion extends Question{

}
class KQuestion extends Question{

}

这是选择题和填空题的类,直接继承了普通题。

ZQuestion q=new ZQuestion();
String str = a.substring(a.indexOf("#Z:") + 3, a.indexOf("#Q:"));
str = str.strip();
q.setNum(Integer.parseInt(str));
q.setQue(a.substring(a.indexOf("#Q:") + 3, a.indexOf("#A:")));
q.setQue(q.getQue().strip());
q.setAns(a.substring(a.indexOf("#A:") + 3));
q.setAns(q.getAns().strip());
q.setExist(true);
ZQue.add(q);

KQuestion q=new KQuestion();
String str = a.substring(a.indexOf("#K:") + 3, a.indexOf("#Q:"));
str = str.strip();
q.setNum(Integer.parseInt(str));
q.setQue(a.substring(a.indexOf("#Q:") + 3, a.indexOf("#A:")));
q.setQue(q.getQue().strip());
q.setAns(a.substring(a.indexOf("#A:") + 3));
q.setAns(q.getAns().strip());
q.setExist(true);
KQue.add(q);

这是选择题和填空题的题目保存,和普通题一样的。

for(int i=0;i<Que.length;i++)
{
    if(Que[i].getNum() ==Del.get(j).getNum())
        Que[i].setExist(false);
}
for(int i=0;i<ZQue.size();i++)
{
    if(ZQue.get(i).getNum()==Del.get(j).getNum())
        ZQue.get(i).setExist(false);
}
for(int i=0;i<KQue.size();i++)
{
    if(KQue.get(i).getNum()==Del.get(j).getNum())
        KQue.get(i).setExist(false);
}

这是将题库里要删除的题目删除。

if(Que[z].getAns().compareTo(Ans[i].getAns()[j])==0&& Que[z].isExist())
{
    Ans[i].getPoint()[j]+= Tes[x].que[Ans[i].getAnu()[j] - 1].getPoint();
    Ans[i].getCheck()[j]= String.valueOf(true);
    Ans[i].setSum(Ans[i].getSum()+ Tes[x].que[Ans[i].getAnu()[j] - 1].getPoint());
}

if(ZQue.get(z).getNum() == Tes[x].que[Ans[i].getAnu()[j] - 1].getNum()&&ZQue.get(z).getExist())
{
    String tres=Ans[i].getAns()[j].strip();
    String[] ans=ZQue.get(z).getAns().split(" ");
    String[] res=tres.split(" ");
    int t=0;
    boolean rig=true;
    for(int k=0;k < res.length&&rig;k++) {
        for(int l=0;l < ans.length;l++){
            if(res[k].equals(ans[l])){
                t++;
                break;
            }
            else if(l == ans.length-1){
                rig=false;
            }
        }
    }
    if(rig){
        if(t==ans.length){
            Ans[i].getPoint()[j]+= Tes[x].que[Ans[i].getAnu()[j] - 1].getPoint();
            Ans[i].getCheck()[j]= String.valueOf(true);
            Ans[i].setSum(Ans[i].getSum()+ Tes[x].que[Ans[i].getAnu()[j] - 1].getPoint());
        }
        else {
            Ans[i].getPoint()[j]+= Tes[x].que[Ans[i].getAnu()[j] - 1].getPoint()/2;
            Ans[i].getCheck()[j]= "partially correct";
            Ans[i].setSum(Ans[i].getSum()+ Tes[x].que[Ans[i].getAnu()[j] - 1].getPoint()/2);
        }
    }
}

if(KQue.get(z).getNum() == Tes[x].que[Ans[i].getAnu()[j] - 1].getNum()&&KQue.get(z).getExist())
{
    String tres=Ans[i].getAns()[j].strip();
    String[] ans=KQue.get(z).getAns().split("或");
    String[] res=tres.split("或");
    int t=0;
    boolean rig=true;
    for(int k=0;k < res.length&&rig;k++) {
        for(int l=0;l < ans.length;l++){
            if(res[k].equals(ans[l])){
                t++;
                break;
            }
            else if(l == ans.length-1){
                rig=false;
            }
        }
    }
    if(rig){
        if(t==ans.length){
            Ans[i].getPoint()[j]+= Tes[x].que[Ans[i].getAnu()[j] - 1].getPoint();
            Ans[i].getCheck()[j]= String.valueOf(true);
            Ans[i].setSum(Ans[i].getSum()+ Tes[x].que[Ans[i].getAnu()[j] - 1].getPoint());
        }
        else {
            Ans[i].getPoint()[j]+= Tes[x].que[Ans[i].getAnu()[j] - 1].getPoint()/2;
            Ans[i].getCheck()[j]= "partially correct";
            Ans[i].setSum(Ans[i].getSum()+ Tes[x].que[Ans[i].getAnu()[j] - 1].getPoint()/2);
        }
    }
}

这是对答卷的批改,统分。选择题和填空题当出现多个答案时,选择题是按空格分开,填空题是按或分开,所以用split分割就行。然后再判断答案的正误,我是用了一个计数器来统计答对的个数,只要有一个答错就是零分,如果没有错误的答案,拿就看答对的个数和正确答案的个数是否相等,相等就是满分,小于正确答案个数就是一半的分,但是这个程序的最后得分没有小数,得的一半分会向下取整。

for (int y = 0; y < Tes[x].que.length; y++) {
    for(int j = 0; j< Ans[i].getAns().length; j++)
    {
        if(Ans[i].getAnu()[j]==(y+1))
        {
            boolean exist=false;
            for(int z=0;z<Que.length;z++)
            {
                if(Que[z].getNum() == Tes[x].que[y].getNum())
                {
                    exist=true;
                    if(Que[z].isExist())
                    {
                        System.out.println(Que[z].getQue() + "~" + Ans[i].getAns()[j] + "~" + Ans[i].getCheck()[j]);
                        break;
                    }
                    else {
                        System.out.println("the question " + Tes[x].que[y].getNum() + " invalid~0");
                        break;
                    }
                }
            };
            for(int z=0;z<ZQue.size();z++){
                if(ZQue.get(z).getNum() == Tes[x].que[y].getNum())
                {
                    exist=true;
                    if(ZQue.get(z).isExist())
                    {
                        System.out.println(ZQue.get(z).getQue() + "~" + Ans[i].getAns()[j] + "~" + Ans[i].getCheck()[j]);
                        break;
                    }
                    else {
                        System.out.println("the question " + Tes[x].que[y].getNum() + " invalid~0");
                        break;
                    }
                }
            }
            for(int z=0;z<KQue.size();z++){
                if(KQue.get(z).getNum() == Tes[x].que[y].getNum())
                {
                    exist=true;
                    if(KQue.get(z).isExist())
                    {
                        System.out.println(KQue.get(z).getQue() + "~" + Ans[i].getAns()[j] + "~" + Ans[i].getCheck()[j]);
                        break;
                    }
                    else {
                        System.out.println("the question " + Tes[x].que[y].getNum() + " invalid~0");
                        break;
                    }
                }
            }
            if(!exist)
                System.out.println("non-existent question~0");
            break;
        }
        else if(j== Ans[i].getAns().length-1)
            System.out.println("answer is null");
    }
}
boolean cout=false;
for(int j=0;j<Stu.length;j++)
{
    if(Stu[j].getSid().compareTo(Ans[i].getSid())==0) {
        System.out.printf("%s %s: ", Stu[j].getSid(), Stu[j].getName());
        break;
    }
    else if(j==Stu.length-1) {
        System.out.println(Ans[i].getSid() + " not found");
        cout=true;
    }
}
if(cout)
    break;
for (int y = 0; y < Tes[x].que.length; y++) {
    for(int j = 0; j< Ans[i].getAns().length; j++) {
        if (Ans[i].getAnu()[j] == (y+1)) {
            System.out.printf("%d", Ans[i].getPoint()[j]);
            if (y < Tes[x].que.length - 1)
                System.out.printf(" ");
            else
                System.out.printf("~");
            break;
        }
        else if(j== Ans[i].getAns().length-1)
        {
            System.out.printf("0");
            if (y < Tes[x].que.length - 1)
                System.out.printf(" ");
            else
                System.out.printf("~");
        }
    }
}
System.out.printf("%d\n", Ans[i].getSum());
break;

最后输出的输出和之前的差不多。就是要分别遍历普通题,选择题,填空题的题库,找到相对应的题目。

  • [ 2 ] 家居强电电路模拟程序-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。

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]
      #T2:[VCC T1-1] [T1-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]
#L1:1.00
end
输出样例10:
在这里给出相应的输出。例如:

@L1:1.00
@R2:180
类图设计:

class Electric  implements Comparable<Electric> {
    private String name;
    private boolean state;
    HashMap<String,Electric> pin1 = new HashMap<>();
    HashMap<String,Electric> pin2 = new HashMap<>();
    private double inVolt=0;
    private double outVolt=0;
    private double PD=inVolt-outVolt;
    public Electric() {

    }
    public double getPD() {
        return PD;
    }
    public void setPD(double PD) {
        this.PD = PD;
    }
    public boolean isState() {
        return state;
    }
    public void setState(boolean state) {
        this.state = state;
    }
    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public double getInVolt() {
        return inVolt;
    }

    public void setInVolt(double inVolt) {
        this.inVolt = inVolt;
    }

    public double getOutVolt() {
        return outVolt;
    }

    public void setOutVolt(double outVolt) {
        this.outVolt = outVolt;
    }

    Electric(String name){
        this.name=name;
    }
    void set(){
    }
    void set(String state){}
    void show(){}

    @Override
    public int compareTo(Electric o) {
        String[] str={"K","F","L","B","R","D"};
        for(int i=0;i<str.length;i++)
        {
            if(this.name.startsWith(str[i])){
                for(int j=0;j<str.length;j++)
                {
                    if(o.getName().startsWith(str[j])) {
                        if (i < j)
                            return -1;
                        else if (i == j)
                            return this.name.compareTo(o.getName());
                        else
                            return 1;
                    }
                }
            }
        }
        return 0;
    }
}

这是电器类的设计,有名字,状态,输入电压,输出电压,电压差,输入引脚和输出引脚以及一个排序方法。这个排序是在最后输出各个电器的时候要的排序,是按照开关、分档调速器、连续调速器、白炽灯、日光灯、吊扇的顺序依次输出所有设备,因为每个电器设备的名字都有特定的格式,以特定的字母开头,我就利用了这个特性,先将这些字母按输出顺序放在字符串数组里,然后按其在字符串数组里的位置排序。

void set(String a){
        if(super.isState()){
            super.setState(false);
            super.setOutVolt(0);
            super.setPD(super.getOutVolt()-super.getInVolt());
        }
        else{
            super.setOutVolt(super.getInVolt());
            super.setPD(super.getOutVolt()-super.getInVolt());
            super.setState(true);
        }
    }
    void show(){
        if(super.isState())
            System.out.println("@"+super.getName()+":closed");
        else
            System.out.println("@"+super.getName()+":turned on");
    }
void set(String state){
        if(state.equals("+")){
            if(position<3)
                position++;
            super.setOutVolt(super.getInVolt()*gear[position]);
            super.setPD(super.getOutVolt()-super.getInVolt());
        }
        else if(state.equals("-")){
            if(position>0)
                position--;
            super.setOutVolt(super.getInVolt()*gear[position]);
        }
    }
    void show(){
        System.out.println("@"+super.getName()+":"+position);
    }
 void set(String state){
        position=Double.parseDouble(state);
        super.setOutVolt(super.getInVolt()*position);
        super.setPD(super.getOutVolt()-super.getInVolt());
    }
    void show(){
        System.out.println("@"+super.getName()+":"+String.format("%.2f",position));
    }
void set(){
        if(super.getPD()<10)
            luminance=0;
        else if(super.getPD()>=10&&super.getPD()<220)
            luminance=50+(super.getPD()-10)*(5.0/7);
        else
            luminance=200;
    }
    void show(){
        System.out.println("@"+super.getName()+":"+(int)luminance);
    }
void set(){
        if(super.getPD()==0)
            luminance=0;
        else
            luminance=180;
    }
    void show(){
        System.out.println("@"+super.getName()+":"+(int)luminance);
    }
void set(){
        if(super.getPD()<80)
            speed=0;
        else if(super.getPD()<=150)
            speed=80+(super.getPD()-80)*4;
        else
            speed=360;
    }
    void show(){
        System.out.println("@"+super.getName()+":"+(int)speed);
    }

每一个设备都有其自己的set()和show()方法。

class ElementLine extends Electric{
    LinkedList<Electric> ele = new LinkedList<>();
    boolean pass;
    ElementLine() {}
    ElementLine(String name) {
        super(name);
    }
}

还有一个串联电路类,并有一个boolean类型的变量来判断这条电路是否是通路。

static ArrayList<Electric> electric = new ArrayList<>();
static ElementLine elementLine = new ElementLine();

在main方法前我先设置了一个Electric的ArryList来保存每一个电器设备,而且在这里排序和输出,并设置了串联电路,以便后面计算各个设备的分压。

String[] a = str.split(" ");
int volt = 0;
for(int i=0;i<a.length;i++)
{
    if(a[i].equals("VCC")){
        volt=220;
        continue;
    }
    else if(a[i].equals("GND")){
        continue;
    }
    else
    {
        String[] element = a[i].split("-");
        if(element[1].equals("1")) {
            Electric elec=null;
            if(element[0].startsWith("K")){
                elec=new KSwitch(element[0]);
            }else if(element[0].startsWith("F")){
                elec=new FGovernor(element[0]);
            }
            else if(element[0].startsWith("L")){
                elec=new LGovernor(element[0]);
            }
            else if(element[0].startsWith("B")){
                elec=new BLamp(element[0]);
            }
            else if(element[0].startsWith("R")){
                elec=new RLamp(element[0]);
            }
            else if(element[0].startsWith("D")){
                elec=new DFan(element[0]);
            }
            if(volt==220){
                elec.setInVolt(220);
                volt=0;
            }
            electric.add(elec);
            elementLine.ele.add(elec);
        }
    }
}

这是处理每一个设备连接信息。由于每次只输入两个引脚,只需要用split把两个引脚分开,当是引脚1的就new一个新设备,根据名字判断是哪种设备。再将设备放入设备数组里和串联电路里。因为是同一个对象,只要修改其中一个,那么这两个里的设备都会被修改。

static void K(String str){
    for(int i=0;i<electric.size();i++){
        if(electric.get(i).getName().equals(str)){
            electric.get(i).set("开关");
        }
    }
}

static void F(String str){
    String name="";
    String state="";
    if(str.contains("+")){
        name=str.substring(0,str.indexOf("+"));
        state="+";
    }
    if(str.contains("-")){
        name=str.substring(0,str.indexOf("-"));
        state="-";
    }
    for(int i=0;i<electric.size();i++){
        if(electric.get(i).getName().equals(name)){
            electric.get(i).set(state);
        }
    }
}
static void L(String str){
    String name=str.substring(0,str.indexOf(":"));
    String state=str.substring(str.indexOf(":")+1);
    for(int i=0;i<electric.size();i++){
        if(electric.get(i).getName().equals(name)){
            electric.get(i).set(state);
        }
    }
}

这是处理控制设备调节信息,调用的是设备的set()方法。

for(int i=0;i<elementLine.ele.size();i++){
  if(i==0) {
      if (elementLine.ele.get(i).getName().startsWith("K") || elementLine.ele.get(i).getName().startsWith("F") || elementLine.ele.get(i).getName().startsWith("L")) {
          continue;
      }else {
          elementLine.ele.get(i).setPD(elementLine.ele.get(i).getInVolt()-elementLine.ele.get(i).getOutVolt());
          elementLine.ele.get(i).set();
      }
  }else{
      elementLine.ele.get(i).setInVolt(elementLine.ele.get(i-1).getOutVolt());
      elementLine.ele.get(i).setPD(elementLine.ele.get(i).getInVolt()-elementLine.ele.get(i).getOutVolt());
      elementLine.ele.get(i).set();
  }
}

这是计算设备的分压,并通过设备的set()方法将设备的参数计算出来。

int cn=0;
int t=0;
for(int i=0;i<elementLine.ele.size();i++){
    if (elementLine.ele.get(i).getName().startsWith("K")){
        cn++;
        if(elementLine.ele.get(i).isState())
            t++;
    }

}
if(cn==t)
    elementLine.setState(true);

判断是否为通路

Collections.sort(electric);
for(int i=0;i<electric.size();i++){
    if (electric.get(i).getName().startsWith("K") || elementLine.ele.get(i).getName().startsWith("F") || elementLine.ele.get(i).getName().startsWith("L")){
        electric.get(i).show();
    }
    else{
        if(elementLine.isState()){
            electric.get(i).show();
        }
        else{
            System.out.println("@"+ electric.get(i).getName()+":"+0);
        }
    }
}

将设备排序并输出设备信息。

  • [ 3 ] 家居强电电路模拟程序-2
    题目要求:智能家居是在当下家庭中越来越流行的一种配置方案,它通过物联网技术将家中的各种设备(如音视频设备、照明系统、窗帘控制、空调控制、安防系统、数字影院系统、影音服务器、影柜系统、网络家电等)连接到一起,提供家电控制、照明控制、电话远程控制、室内外遥控、防盗报警、环境监测、暖通控制、红外转发以及可编程定时控制等多种功能和手段。与普通家居相比,智能家居不仅具有传统的居住功能,兼备建筑、网络通信、信息家电、设备自动化,提供全方位的信息交互功能。请根据如下要去设计一个智能家居强电电路模拟系统。以下题目介绍中加粗的部分为本次迭代在“家居强电电路模拟程序-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
类图设计:

void set(){
        if(super.getPD()<80)
            speed=0;
        else if(super.getPD()<=100)
            speed=80;
        else if(super.getPD()<=120)
            speed=160;
        else if(super.getPD()<=140)
            speed=260;
        else
            speed=360;
    }
    void show(){
        System.out.println("@"+super.getName()+":"+(int)speed);
    }

这次新增了落地扇,这是其set()和show()方法。

String[] str={"K","F","L","B","R","D","A"};

排序只需要在原来的数组后面加上落地扇的命名开头字母。

class ParallelLine extends Electric{
    ArrayList<ElementLine> TLine = new ArrayList<>();
    ParallelLine(){}
    ParallelLine(String name){super(name);}
}

新加了一个并联电路类

static ArrayList<ElementLine> TLine = new ArrayList<>();
static ArrayList<ParallelLine> MLine=new ArrayList<>();

在上一次的基础上新设置了一个串联电路各分路和并联电路

static void saveElement(String str){
    String name=str.substring(0,str.indexOf(":"));
    ElementLine eleLine=null;
    str=str.substring(str.indexOf(":")+1);
    String pat = "\\[(.+?)\\]";
    Pattern pattern = Pattern.compile(pat);
    Matcher m = pattern.matcher(str);
    if(name.startsWith("T")) {
        boolean mainLine = false;
        while (m.find()) {
            String[] a = m.group(1).split(" ");
            for (int i = 0; i < a.length; i++) {
                if (a[i].equals("IN")) {
                    eleLine = new ElementLine(name);
                    TLine.add(eleLine);
                    continue;
                } else if (a[i].equals("OUT")) {
                    continue;
                } else if (a[i].equals("VCC")) {
                    mainLine = true;
                    elementLine = new ElementLine(name);
                    continue;
                } else if (a[i].equals("GND")) {
                    continue;
                } else {
                    String[] st = a[i].split("-");
                    if (st[1].equals("1")) {
                        Electric elec = null;
                        if (st[0].startsWith("K")) {
                            elec = new KSwitch(st[0]);
                        } else if (st[0].startsWith("F")) {
                            elec = new FGovernor(st[0]);
                        } else if (st[0].startsWith("L")) {
                            elec = new LGovernor(st[0]);
                        } else if (st[0].startsWith("B")) {
                            elec = new BLamp(st[0]);
                        } else if (st[0].startsWith("R")) {
                            elec = new RLamp(st[0]);
                        } else if (st[0].startsWith("D")) {
                            elec = new DFan(st[0]);
                        } else if (st[0].startsWith("A")) {
                            elec = new AFan(st[0]);
                        }
                        electric.add(elec);
                        if (mainLine) {
                            elementLine.ele.add(elec);
                            elementLine.setResistance(elementLine.getResistance() + elec.getResistance());
                        } else {
                            eleLine.ele.add(elec);
                            eleLine.setResistance(eleLine.getResistance() + elec.getResistance());
                        }
                    }
                    if (st[1].equals("IN")) {
                        elementLine.ele.add(new Electric(st[0]));
                    }
                }
            }
        }
    }
    if(name.startsWith("M")){
        MLine.add(new ParallelLine(name));
        while(m.find())
        {
            String[] a=m.group(1).split(" ");
            for(int i=0;i<a.length;i++)
            {
                for(int j=0;j<MLine.size();j++)
                {
                    if(name.equals(MLine.get(j).getName())){
                        for(int x=0;x<TLine.size();x++){
                            if(TLine.get(x).getName().equals(a[i])){
                                MLine.get(j).TLine.add(TLine.get(x));
                            }
                        }
                    }
                }
            }
        }
    }
}

这是对每条设备连接信息的处理,分了是以T开头还是M开头。如果以T开头,通过是含"IN"还是"VCC"来判断是主路还是分路。在主路中还会穿插并联电路,并联电路的引脚是"IN",所以还要判断会不会有引脚是"IN"的。如果以M开头就要将输入的串联电路放入并联电路里。

static void K(String str){
  for(int i=0;i<electric.size();i++){
      if(electric.get(i).getName().equals(str)){
          electric.get(i).set("开关");
      }
  }
}
static void F(String str){
  String name="";
  String state="";
  if(str.contains("+")){
      name=str.substring(0,str.indexOf("+"));
      state="+";
  }
  if(str.contains("-")){
      name=str.substring(0,str.indexOf("-"));
      state="-";
  }
  for(int i=0;i<electric.size();i++){
      if(electric.get(i).getName().equals(name)){
          electric.get(i).set(state);
      }
  }
}
static void L(String str){
  String name=str.substring(0,str.indexOf(":"));
  String state=str.substring(str.indexOf(":")+1);
  for(int i=0;i<electric.size();i++){
      if(electric.get(i).getName().equals(name)){
          electric.get(i).set(state);
      }
  }
}

控制设备调节信息和之前的一样。

if(elementLine.ele.get(i).getName().startsWith("M")){
    for(int j=0;j<MLine.size();j++){
        if(MLine.get(j).getName().equals(elementLine.ele.get(i).getName())){
            for(int x=0;x<MLine.get(j).TLine.size();x++){
                for(int n=0;n<MLine.get(j).TLine.get(x).ele.size();n++)
                {
                    if(MLine.get(j).TLine.get(x).ele.get(n).getName().startsWith("K"))
                    {
                        MLine.get(j).TLine.get(x).setCnOfK(MLine.get(j).TLine.get(x).getCnOfK()+1);
                        if(MLine.get(j).TLine.get(x).ele.get(n).isState())
                            MLine.get(j).TLine.get(x).setCnOfONK(MLine.get(j).TLine.get(x).getCnOfONK()+1);
                    }
                }
                if(MLine.get(j).TLine.get(x).getCnOfK()==MLine.get(j).TLine.get(x).getCnOfONK()) {
                    MLine.get(j).TLine.get(x).setState(true);
                    MLine.get(j).setState(true);
                    MLine.get(j).setResistance(MLine.get(j).getResistance()+1/MLine.get(j).TLine.get(x).getResistance());
                }
            }
            elementLine.setCnOfK(elementLine.getCnOfK()+1);
            if(MLine.get(j).isState()) {
                elementLine.setCnOfONK(elementLine.getCnOfONK()+1);
                MLine.get(j).setResistance(1 / MLine.get(j).getResistance());
                elementLine.setResistance(elementLine.getResistance() + MLine.get(j).getResistance());
            }
        }
    }
}
if(elementLine.ele.get(i).getName().startsWith("K")){
    elementLine.setCnOfK(elementLine.getCnOfK()+1);
    if(elementLine.ele.get(i).isState()){
        elementLine.setCnOfONK(elementLine.getCnOfONK()+1);
    }
}

这是计算每条并联电路的电阻,并将电阻加到主电路里面,在计算电阻时就将每条串联电路是否是通路判断出来。

if(elementLine.getCnOfK() == elementLine.getCnOfONK()) {
    elementLine.setState(true);
    if(elementLine.ele.get(0).getName().startsWith("F")||elementLine.ele.get(0).getName().startsWith("L"))
        elementLine.setPD(220*elementLine.ele.get(0).getPD());
    else
        elementLine.setPD(220);
    elementLine.setElectricity(elementLine.getPD()/elementLine.getResistance());
}else {
    elementLine.setElectricity(0);
}

计算出主路的电流。

if(elementLine.ele.get(i).getName().startsWith("K") || elementLine.ele.get(i).getName().startsWith("F") || elementLine.ele.get(i).getName().startsWith("L"))
    continue;
else if(elementLine.ele.get(i).getName().startsWith("M")){
    for(int j=0;j<MLine.size();j++){
        if(MLine.get(j).getName().equals(elementLine.ele.get(i).getName())){
            MLine.get(j).setPD(elementLine.getElectricity()*MLine.get(j).getResistance());
            for(int x=0;x<MLine.get(j).TLine.size();x++){
                if(MLine.get(j).TLine.get(x).isState()) {
                    MLine.get(j).TLine.get(x).setElectricity(MLine.get(j).getPD()/MLine.get(j).TLine.get(x).getResistance());
                    for (int n = 0; n < MLine.get(j).TLine.get(x).ele.size(); n++){
                        MLine.get(j).TLine.get(x).ele.get(n).setPD(MLine.get(j).TLine.get(x).getElectricity()*MLine.get(j).TLine.get(x).ele.get(n).getResistance());
                        MLine.get(j).TLine.get(x).ele.get(n).set();
                    }
                }
            }
        }
    }
}else {
    elementLine.ele.get(i).setPD(elementLine.getElectricity()*elementLine.ele.get(i).getResistance());
    elementLine.ele.get(i).set();
}

这是计算设备的分压,并通过设备的set()方法将设备的参数计算出来。

Collections.sort(electric);
for(int i=0;i<electric.size();i++){
    if (electric.get(i).getName().startsWith("K") || electric.get(i).getName().startsWith("F") || electric.get(i).getName().startsWith("L")){
        electric.get(i).show();
    }
    else{
        if(elementLine.isState()){
            electric.get(i).show();
        }
        else{
            System.out.println("@"+ electric.get(i).getName()+":"+0);
        }
    }
}

将设备排序并输出设备信息。
3. 踩坑心得

  • [ 1 ] 这次的家居强电电路模拟程序在每次发布前会给出初稿让我们设计,我会根据题目后在脑子里构思但是没有写代码去实现,导致我会想得比较多,有些设计都不是很合理,而且还会忘记了。就像每个设备都会有引脚。我当时是想这这些设备通过引脚连接的,不就得有引脚嘛。但是最后在写的过程中就感觉直接在串联电路里连起来,然后在串联电路里分析就行了。所以在写程序的时候不能只想,还要根据实际,一边设计一边把思考写下来,这样才是有用的思考。
  • [ 2 ] 在写代码的时候一定要仔细分析。这次设计的分档开关一开始是错的,因为他是有上限和下限的,而我的上限没设置好,导致了当超过了挡位就错了。
    下面是刚开始的
void set(String state){
  if(state.equals("+")){
      if(position<4)
          position++;
      super.setOutVolt(super.getInVolt()*gear[position]);
      super.setPD(super.getOutVolt()-super.getInVolt());
  }
  else if(state.equals("-")){
      if(position>0)
          position--;
      super.setOutVolt(super.getInVolt()*gear[position]);
  }
}

这是改正后的

void set(String state){
if(state.equals("+")){
    if(position<3)
        position++;
    super.setOutVolt(super.getInVolt()*gear[position]);
    super.setPD(super.getOutVolt()-super.getInVolt());
}
else if(state.equals("-")){
    if(position>0)
        position--;
    super.setOutVolt(super.getInVolt()*gear[position]);
}
}
  • [ 3 ] 在计算分压的时候没有考虑到第一个不是开关或者是调速器,导致有非零返回。
    这是之前的
if(elementLine.ele.get(i).getName().startsWith("K")||elementLine.ele.get(i).getName().startsWith("F")||elementLine.ele.get(i).getName().startsWith("L")){
    continue;
}else{
    elementLine.ele.get(i).setInVolt(elementLine.ele.get(i-1).getOutVolt());
    elementLine.ele.get(i).setPD(elementLine.ele.get(i).getInVolt()-elementLine.ele.get(i).getOutVolt());
    elementLine.ele.get(i).set();
}

这是改正之后的

if(i==0) {
    if (elementLine.ele.get(i).getName().startsWith("K") || elementLine.ele.get(i).getName().startsWith("F") || elementLine.ele.get(i).getName().startsWith("L")) {
        continue;
    }else {
        elementLine.ele.get(i).setPD(elementLine.ele.get(i).getInVolt()-elementLine.ele.get(i).getOutVolt());
        elementLine.ele.get(i).set();
    }
}else{
    elementLine.ele.get(i).setInVolt(elementLine.ele.get(i-1).getOutVolt());
    elementLine.ele.get(i).setPD(elementLine.ele.get(i).getInVolt()-elementLine.ele.get(i).getOutVolt());
    elementLine.ele.get(i).set();
}
  • [ 4 ] 题目要求了只有在最后输出时再把计算结果按截尾规则,舍弃尾数,保留整数输出,但是按照我的计算方法当我的计算结果是6.9999999999999999时我的结果是6,但是正确的答案应该是7.这个可能的在计算的过程的差异导致了我计算出的数值出错了
    原来是直接把double类型转换成int类型
void show(){
    System.out.println("@"+super.getName()+":"+(int)luminance);
}

改正之后

void show(){
    double df1=Math.ceil(luminance)-luminance;
    double df2=luminance-Math.floor(luminance);
    if(df1<1e-10||df2<1e-10)
        luminance=Math.round(luminance);
    System.out.println("@"+super.getName()+":"+(int)luminance);
}
  1. 总结
  • 通过这三次题目集,我更深刻的理解了类的继承和多态,也能运用类的这个特性。但是类的多态的使用还是不那么灵活,还需要更深入的学习。
  • 在发布PTA的初稿的时候,要是有多一点的输入样例会更好地让我们提前设计程序。

标签:总结,输出,题目,引脚,PTA,电路,输入,设备
From: https://www.cnblogs.com/theBlogForStudy/p/18233656

相关文章

  • PTA训练集阶段总结blog
    目录PTA训练集总结blog1.前言2.设计与分析题目集一7.4答题判题程序四关于设计要求:UML类图及设计分析:部分源码:复杂度分析:题目集五7.1家具强电电路模拟系统—1关于设计要求:UML类图及设计分析:部分源码:复杂度分析:题目集六家居强电电路模拟程序-2关于设计要求:UML类图及设计分析:部......
  • 两次pta题目总结
    两次pta题目总结写在前面这两次pta是新题目,上次那个答题判题的没有继续迭代了,根据这两次题目我也学到了一些东西,主要是面向对象设计模式的运用,接口的使用,以及递归算法的使用等等第一题题目内容:智能家居是在当下家庭中越来越流行的一种配置方案,它通过物联网技术将家中的各种设......
  • 南昌航空大学pta总结
    前言背景介绍:继上次PTA题目讲解完,本次博客会进行对后几次题目集的讲解说明和题后总结目的和目标:及时反馈学习成果,达到学由所会,及时吸收和深入学习的目的学习收益:通过后面几次的题目学习,使得我对java的语法代码以及运用熟练度有了极大的进步,同时也对学习代码语言有了一个更......
  • abc--cf训练日常总结
    ABC最近遇到好多思维和位运算的题目不会做,特地过来总结一些小小的知识点。思维题目https://atcoder.jp/contests/abc353/tasks/abc353_c这道题目要求我们计算连续的两个相邻的数组元素之和。我一开始用暴力,后面换了种错误的思路就wa了。其实这道题目是求和,然后找到和大于1e8......
  • 栈经典题目(C++)
    文章目录前言一、删除字符串中的所有相邻重复项1.题目解析2.算法原理3.代码编写二、基本计算器II1.题目解析2.算法原理3.代码编写三、字符串解码1.题目解析2.算法原理3.代码编写四、验证栈序列1.题目解析2.算法原理3.代码编写总结前言一、删除字符串中的所有......
  • 题目集4-6的总结性Blog
    一,前言一-1.第四次题集是接着前面的第三次题集的迭代,知识点主要是继承的运用,正则表达式的运用同样少不了,相对于前面一次添加了对选题以及填空题,难度也相对于前一次加大。一-2.第五次题集是新的题集迭代是有关电路的,知识点有抽象类,抽象方法的使用以及继承的使用,同样正则表达式;由......
  • 【机器学习基础】Python编程07:五个实用练习题的解析与总结
    Python是一种广泛使用的高级编程语言,它在机器学习领域中的重要性主要体现在以下几个方面:简洁易学:Python语法简洁清晰,易于学习,使得初学者能够快速上手机器学习项目。丰富的库支持:Python拥有大量的机器学习库,如scikit-learn、TensorFlow、Keras和PyTorch等,这些库提供了......
  • 【机器学习基础】Python编程08:五个实用练习题的解析与总结
    Python是一种广泛使用的高级编程语言,它在机器学习领域中的重要性主要体现在以下几个方面:简洁易学:Python语法简洁清晰,易于学习,使得初学者能够快速上手机器学习项目。丰富的库支持:Python拥有大量的机器学习库,如scikit-learn、TensorFlow、Keras和PyTorch等,这些库提供了......
  • Activity活动详简总结
    简介        活动(Activity是一种可以包含用户界面的组件主要用于和用户进行交互。一、创建项目NewProject    选择NoActivity,创建一个空活动:     语言选择Java,点击Finish,等待工程加载二、MainActivity活动类   找到如下图路径: ......
  • NSSCTF-Web题目6
    目录[NISACTF2022]checkin1、题目2、知识点3、思路[NISACTF2022]babyupload1、题目2、知识点3、思路[SWPUCTF2022新生赛]1z_unserialize1、题目2、知识点3、思路[NISACTF2022]checkin1、题目2、知识点010编辑器的使用、url编码3、思路打开题目,得到源......