OO作业总结(1-3)
前言
这三次作业,是我第一次,真正的系统性接触和学习Java,在这三次作业中,我对Java的理解也逐渐加深,也逐渐领悟到了从C语言入门的我在应对工程代码方面的不足,这一次学习到的度量分析又为我打开了新的大门。
(一)代码的度量分析
(1)第一次作业
第一次作业,内容是多项式计算,其中并没有涉及到复杂的计算原理和程序逻辑,而是在输入和输出的处理上略有难度。对于当时不懂正则表达式的我来说,我的第一次作业使用了状态自动机。
在第一次作业中,分为3个类,Input类处理输入,Polynomial作为主类,Term类表示单个多项式。之前我一直觉得这份代码写得还算不错,直到我最近学习了Java代码的度量分析,才惊觉之前的代码有着诸多缺点。
接下来贴上分析结果截图
显然,我的循环复杂度(Cyclomatic complexity)在Input类里过高了,也就是说我的Input类功能不够单一,做不到功能内聚,需要再划分为更小的模块,才能减小程序的漏洞数。
这次由于将状态机封装进了Input类,导致的Input类复杂度过高,以后的状态机或许可以分成多个简单类的交互。
(2)第二次作业
第二次作业,内容是傻瓜电梯的调度。虽然指导书刻意复杂化了代码逻辑,但其实代码非常之简单,只需要每次处理当前请求时,将请求队列之中的同质请求消除就可以了。
在这一次作业当中,我也意识到了try,catch的重要性,在面临可能数组越界或是访问空指针的危险情况时,try and catch变得尤为重要,这样可以快速帮助锁定错误所在,希望以后我也能像各位大牛一样将try catch运用熟练。
下面贴上类图和度量分析结果
本次代码设计上的主要问题从度量分析看来主要来自于Request类的SelfCheck方法,但是本人点进去仔细查看SelfCheck函数后却并不认为其有什么问题,代码如下:
1 boolean SelfCheck(double LastTime)
2 {
3 if(Time<LastTime)return false;
4 if(Type==0 && (Aim==0 || Aim>10))return false;
5 if(Type==1 && (Dep<=1 || Dep>10))return false;
6 if(Type==2 && (Dep==0 || Dep>9))return false;
7 if(Time>(1L<<32)-1)return false;
8 return true;
9 }
这个功能仅仅用来判断Request类是否合法,而由于限制条件的众多,代码内的众多if语句也难以避免,况且该功能十分明确且简单,个人不认为这段代码有不妥的地方。
以上就是这次分析的结果,个人认为这次的情况正好对应了维基百科上所说的对于特殊情况,代码的循环复杂度可以酌情放宽。
(3)第三次作业
关于第三次作业,我需要提前申明的是,本次作业我一开始的想法就错了,导致后面回天乏术。具体原因会在第二部分的bug互查说明,这里仅仅分析代码的性能。
下面贴上度量分析结果:
下面贴上Watcher中的FindDir作为示例:
1 Request FindDir(double time)
2 {
3 if(time==L.GetLast()+1 && L.IsOpen())
4 {
5 //System.out.println("LastTime="+L.GetLast()+" time="+time);
6 L.Change(time);
7 //System.out.println(L);
8 if(Temp.GetFor()==0)
9 {
10 if(L.GetCnt(L.GetPos())>0 && L.GetReq(L.GetPos(), 1).GetAim()==L.GetPos())
11 {
12 OUT.Add("VALID", 0, L.GetReq(L.GetPos(), 1), L.GetPos(), time);
13 L.pop(L.GetPos());
14 }
15 if(F[L.GetPos()].GetQn()>0 && F[L.GetPos()].GUP(1).GetDep()==L.GetPos())
16 {
17 //System.out.print("First:");F[L.GetPos()].GUP(1).Print();
18 OUT.Add("VALID", 0, F[L.GetPos()].GUP(1), L.GetPos(), time);
19 F[L.GetPos()].PopQ();
20 }
21 if(F[L.GetPos()].GetDn()>0 && F[L.GetPos()].GDN(1).GetDep()==L.GetPos())
22 {
23 OUT.Add("VALID", 0, F[L.GetPos()].GDN(1), L.GetPos(), time);
24 F[L.GetPos()].PopD();
25 }
26 while(L.GetCnt(L.GetPos())>0 && L.GetReq(L.GetPos(), 1).GetAim()==L.GetPos())
27 {
28 OUT.Add("#SAME", 0, L.GetReq(L.GetPos(), 1), L.GetPos(), time);
29 L.pop(L.GetPos());
30 }
31 while(F[L.GetPos()].GetQn()>0 && F[L.GetPos()].GUP(1).GetDep()==L.GetPos())
32 {
33 OUT.Add("#SAME", 0, F[L.GetPos()].GUP(1), L.GetPos(), time);
34 F[L.GetPos()].PopQ();
35 }
36 while(F[L.GetPos()].GetDn()>0 && F[L.GetPos()].GDN(1).GetDep()==L.GetPos())
37 {
38 OUT.Add("#SAME", 0, F[L.GetPos()].GDN(1), L.GetPos(), time);
39 F[L.GetPos()].PopD();
40 }
41 }
42 }
43 //System.out.println("haha="+L.GetAim()+" "+L.GetPos()+" "+L.IsOpen()+" time="+time+" last="+L.GetLast());
44 //L.Print(time);
45 if(L.GetAim()==L.GetPos() && !L.IsOpen())
46 {
47 //System.out.println("hahahahahahahhahaha");
48 //L.Print(time);
49
50 //L.Print(time);
51 int aim1=0, aim2=0, aim3=0;
52 if(time==L.GetLast())
53 {
54 int finish = L.Finish();
55 if(finish!=0)
56 {
57 L.Direct(L.GetAim()-L.GetPos()==0 ? 0 : L.GetAim()-L.GetPos()==1 ? 1 : -1);
58 return new Request(-1);
59 }
60 }
61 //L.Print(time);
62 for(int i=1;i<=10;++i)
63 {
64 if(L.GetCnt(i)>0 && (aim1==0 || L.GetFront(i).GetSeq()<L.GetFront(aim1).GetSeq()))
65 aim1 = i;
66 if(F[i].GetQn()>0 && (aim2==0 || F[i].GUP(1).GetSeq()<F[aim2].GUP(1).GetSeq()))
67 aim2 = i;
68 if(F[i].GetDn()>0 && (aim3==0 || F[i].GDN(1).GetSeq()<F[aim3].GUP(1).GetSeq()))
69 aim3 = i;
70 }
71 if(aim1>0 &&
72 (aim2==0 || (F[aim2].GetQn()>0 && L.GetFront(aim1).GetSeq()<F[aim2].GUP(1).GetSeq())) &&
73 (aim3==0 || (F[aim3].GetDn()>0 && L.GetFront(aim1).GetSeq()<F[aim3].GDN(1).GetSeq()))) return L.GetFront(aim1);
74 if(aim2>0 &&
75 (aim3==0 || (F[aim3].GetDn()>0 && F[aim2].GUP(1).GetSeq()<F[aim3].GDN(1).GetSeq())) &&
76 (aim1==0 || (L.GetCnt(aim1)>0 && L.GetFront(aim1).GetSeq()>F[aim2].GUP(1).GetSeq()))) return F[aim2].GUP(1);
77 if(aim3>0 &&
78 (aim2==0 || (F[aim2].GetQn()>0 && F[aim2].GUP(1).GetSeq()>F[aim3].GDN(1).GetSeq())) &&
79 (aim1==0 || (L.GetCnt(aim1)>0 && L.GetFront(aim1).GetSeq()>F[aim3].GDN(1).GetSeq()))) return F[aim3].GDN(1);
80 return new Request(-1);
81 }
82 //L.Print(time);
83 return new Request();
84 }
可见由于之前的逻辑分析失误,导致了if-else语句的滥用,由此可见,在写代码之前,有一个清晰,简单的框架是写好程序之必备,而本人在这一方面则反了经验不足的错误,以为简单按照描述来写就行,结果表明不成熟的程序框架会导致大量的冗余代码和复杂的·逻辑判断,今后必须引以为戒。
(二)bug分析
(1)第一次作业
第一次作业本人存在一个bug,那就是没有判断程序最开始是空格的情况。因为本人的最初版本程序要求程序第一个左括号,所以空格会导致判断为ERROR。
由此可见两个问题:
1.对内置函数不熟练,本来可以用string.replace来解决
2.逻辑思维不够严密
(2)第二次作业
第二次作业本人由于对助教意思的误解,对于同质请求没有输出#SAME,这是一个incomplete
(3)第三次作业
而第三次作业,本人为了完全再现电梯接受请求的反应,决定按照时间的运行来模拟,而这时由于一开始得考虑不周,本人犯了一些严重的错误:
1.电梯请求放在了电梯类,楼层请求放在了楼层类,导致交互时的代码非常繁琐
2.由于一开始的考虑不周,放弃了请求队列的想法,本意是想每次接受请求时只与当前请求比较,结果发现请求有可能排成队列
3.对于时间和请求的交互请求关系考虑不周,导致出现电梯没有及时改变方向,出现runtime error
综上所述,本人的第三次作业并没有达到理想要求,并非是修补bug可以解决的问题需要推倒重来。
以后也要吸取教训,对待程序一定要思考清楚后开始写,否则仅是浪费时间而已。