简述
这一单元是整个学期的第三单元,第三单元是JML契约式编程。JML语法和高级语言语法类似,学习起来比较容易,上手比较快,参考课程组发布的JML Level 0手册,可以毫无障碍地理解JML的语义。本单元的任务是将助教们书写的JML转化成为自己的程序代码,和课程组下发的官方包共同构成完整的程序。
看似简单的jml,实则蕴藏玄机。照搬代码不可取,强测统统tle。
第一次作业
问题概要
本次作业为实现 person
类和简单社交关系的模拟和查询,学习 JML 规格,构造PersonIdNotFoundException
,EqualPersonIdException
,RelationNotFoundException
,EqualRelationException
的异常类子类处理程序运行时候产生的异常。
设计策略和容器选择
本次作业是jml单元的第一次作业,我的实现方式是完全照搬已有的jml规格进行书写。jml中已经明确地指出了数据规格和方法规格。在理解这些规格之后,我将这些规格使用java语言表达,编写出符合jml语义限制的程序。
在本单元的第一次作业中,我是将所有的代码编写放在了阅读jml之后。在阅读完全部的jml后,我对整个程序的架构有了一定的了解。在完成本次作业的最初阶段,首先选择合适的容器来满足jml的数据规格,之后按照方法规格完成每一个要填充的方法。
在严格的jml约束下,选择合适的容器是本次作业最重要的程序设计环节。合理的容器选择能够为程序构建良好的架构,降低程序运行的时间复杂度,有利于提升程序的可读性和正确性,降低程序运行出错的可能。
在本次的设计中,我选择在MyPerson
中使用两个HashMap,HashMap<Integer, Person> acquaintance
,HashMap<Integer, Integer> value
来存储这个人的所有熟人和对应的value
。因为在MyPerson
类中会有queryValue
和isLinked
查询熟人的方法,如果单单使用Arraylist会让查找的时间复杂度大大提升。HashMap
查找的时间复杂度是O(1),使用HashMap
可以有效降低程序查找的时间复杂度。
在MyNetwork
中我使用了HashMap
类型的容器,HashMap<Integer, Person> people
,来存储所有的人,这样的设计理由也是显而易见的,因为在contains
,getPerson
都涉及查找的操作,因此使用HashMap
可以有效降低程序查找的时间复杂度。
完成了容器的合理选择,整个程序的编写变得十分简单,按照jml的方法规格逐一完成每一个方法即可。
性能问题
本次作业的正确性实现难度不大,jml对于每一个方法的注释杜绝了自然语言的二义性。短短5行的java代码可能需要20多行的jml来解释,助教们严谨的jml语言描述大大降低了本单元的作业难度,近似伪代码的jml语句可以稍作修改成为java程序的一部分。
直接照搬jml语言来实现具有以下优点:降低程序实现的复杂度,程序出错概率大大降低,检查代码仅需要逐条对照jml语言与java语言即可。如果没有cpu运行时间限制的话,这样的实现可以说是完成作业的最稳妥的实现方式。
但是在实际的作业中,这样的实现方式很容易造成程序的时间复杂度过高,程序的可读性下降,严谨的jml语言适合机器来读,直接照搬的话会有判断条件冗长冗余的问题,给他人阅读代码造成一定的障碍。
在本次作业中最容易超时的方法是queryBlockSum
和isCircle
。isCircle
如果用宽度优先搜索的话,时间复杂度是O(E+V)
,这样的时间复杂度在本次作业中不会出现超时的问题。但是queryBlockSum
如果是照搬jml语法来实现的话,需要对所有的人进行二重遍历的话,将会产生O(V^2\*(E+V))
的超级高的时间复杂度,这样的话cpu运行时间很容易超时。
本次作业中queryBlockSum
和isCircle
我都是采用宽度优先搜索算法实现的,在1000条指令2scpu的限制下并没有出现超时的现象。可见本单元第一次作业还是很仁慈的,没有在时间上设置太多障碍,即使没有使用并查集来实现queryBlockSum
和isCircle
,也能运行不超时。
如果容器选择ArrayList
来存储所有的数据的话,会产生查询时间复杂度过大的问题,可能造成cpu超时。对比HashMap
查找的O(1)
的时间复杂度,Arraylist
的查询复杂度是O(n)
。在程序中大部分方法都会去查询容器,降低查询的时间复杂度是拒绝ctle的重要方法之一。
UML类图
作业架构
通过UML类图我们可以看到作业的整体架构就是普通类+异常类。普通类包含MyPerson
和MyNetwork
,其中MyPerson
实现了Person
接口,MyNetwork
实现了Network
接口。异常类包含MyEqualPersonIdException
,MyEqualRelationException
,MyPersonIdNotFoundException
,MyRelationNotFoundException
。MyEqualPersonIdException
继承了EqualPersonIdException
,MyEqualRelationException
继承了EqualRelationException
,MyPersonIdNotFoundException
继承了PersonIdNotFoundException
,MyRelationNotFoundException
继承了RelationNotFoundException
。
在本次的设计中,我选择在MyPerson
中使用两个HashMap
,HashMap<Integer, Person> acquaintance
,HashMap<Integer, Integer> value
来存储这个人的所有熟人和对应的value
。因为在MyPerson
类中会有queryValue
和isLinked
查询熟人的方法,使用HashMap
可以有效降低程序查找的时间复杂度。
在MyGroup
中使用HashMap
来记录组内的所有的人,HashMap<Integer, Person> people
。
在MyNetwork
中我使用了HashMap
类型的容器,HashMap<Integer, Person> people
,来存储所有的人,这样的设计理由也是显而易见的,因为在contains
,getPerson
都涉及查找的操作,因此使用HashMap
可以有效降低程序查找的时间复杂度。
在本次作业中我也没有设置其他的类,所有的类都是严格按照官方包一一对应实现的。不同的类之间的关系同指导书一致。
myBug&&测评与互测
中测通过,强测通过,互测通过
在编写代码的时候发现,前期设计越到位,容器选择得越合理,后期产生的bug会越少。应当按照课程组代码风格来控制每一个方法的长度,这样能有效降低代码出错的概率。
在提交作业之前要进行大量有效的测试,尽可能穷举所有的边界情况。同时可以采用自动测评机的方式对代码进行功能性的测试。当然,设计测评机的过程也是对于题目要求的再一次明确。
在本单元的作业中除了程序的正确性外,程序运行的速度也是十分重要的因素。
在本次作业中,虽然我没有采取并查集来实现isCircle
和queryBlockSum
方法,仅仅使用宽度优先搜索的方式来实现isCircle
和queryBlockSum
,但是在本次作业中并没有出现超时的情况。强测中所有的点都有通过,互测中也没有测试出来超时的问题。
初来第三单元,显然我还没有做好充足的准备,第一次的作业仍旧延续前几个单元的测试方案——仅仅针对程序的正确性进行充分的测试,默认自己的程序在运行时间的方面是正确的。
显然,这样的测试思路在本单元是不适用的,程序很有可能会在强测和互测中运行超时。
在第一次作业完成之后,我完全没有测试程序运行的时间。如何统计程序的运行时间也是在互测同学的代码中学习到的,准备如此不充分,竟然能侥幸通过强测,互测?助教表示(纯属臆想),这周是送分周,下次作业才是这个单元的开始……
在本次互测中同屋同学出现了cpu超时的情况,出现超时的方法是queryBlockSum
方法。很显然,这样的bug我肯定是发现不了的,整个互测的全过程自己也仿佛是旁观者,毫无参与度。
在本次作业中正确性很难出错,同学们出错的地方大多也是没有控制好方法的时间复杂度。相比于前两个单元开放式的设计不同,本次的程序设计开放程度不大,重要的方法由于jml已经命名,所以每一个同学程序的整体架构都是类似的。因此,在本次互测中,白盒测试不失为一种好的测试手段,通过认真研读每一个同学不同方法的实现方案,可以标定出来每一个方法的时间复杂度,可以更有针对性地构造可能产生错误的数据。
此外,发现一个神奇的事情,在第二单元互测时本地超时的数据在网站上可能不超时,但是在第三单元互测的时候发现在本地不超时的数据在网站上可能超时。网站上的程序运行时间具有不确定性。
第二次作业
问题概要
实现一个社交关系模拟系统,通过实现官方提供的接口 Person
、Network
和 Group
,来实现自己的 Person
、Network
和Group
类。通过实现官方提供的接口 Message
,来实现自己的 Message
类。构造EqualGroupIdException
,EqualMessageIdException
,EqualPersonIdException
,EqualRelationException
,GroupIdNotFoundException
,MessageIdNotFoundException
,PersonIdNotFoundException
,RelationNotFoundException
异常类的子类来实现程序中的异常处理。
设计策略和容器选择
本次作业是jml单元的第二次作业,我的实现方式是完全照搬已有的jml规格进行书写。jml中已经明确地指出了数据规格和方法规格。在理解这些规格之后我将这些规格使用java语言表达,编写出符合jml语义限制的程序。
在本单元的第二次作业中,我是将所有的代码编写放在了阅读jml之后。在阅读完全部的jml后,我对整个程序的架构有了一定的了解。在完成本次作业的最初阶段,首先选择合适的容器来满足jml的数据规格,之后按照方法规格完成每一个要填充的方法。
在严格的jml约束下,选择合适的容器是本次作业最重要的程序设计环节。合理的容器选择能够为程序构建良好的架构,降低程序运行的时间复杂度,有利于提升程序的可读性和正确性,降低程序运行出错的可能。
在本次的设计中,在普通类中MyPerson
中有三个容器HashMap<Integer, Person> acquaintance
,HashMap<Integer, Integer> value
,ArrayList<Message> messages
,用来存储熟人队列和熟人之间的value
值和对应的消息对象。在MyNetwork
中拥有4个容器,HashMap<Integer, Person> people
,HashMap<Integer, Group> grouphash
,HashMap<Integer, Message> meshash
,HashMap<Integer, Integer> reps
。people
用来存储社交网络里的所有人,grouphash
用来存储社交网络里面的所有的组,meshash
用来存储社交网络里的所有的消息,reps
则是并查集时候用来记录代表元的哈希字典。
同第一次作业一样,因为容器内部的元素无需有序,并且对容器中元素的查找比较频繁,所以选择HashMap
容器来存储数据。
完成了容器的合理选择,那么整个程序的编写变得十分简单,按照jml的方法规格逐一完成每一个方法即可。
UML类图
作业架构
通过UML类图我们可以看到作业的整体架构就是普通类+异常类。普通类包含MyPerson
,MyNetwork
,MyMessage
和MyGroup
,其中MyPerson
实现了Person
接口,MyNetwork
实现了Network
接口,MyMessage
实现了Message
接口,MyGroup
实现了Group
接口。异常类包含MyEqualPersonIdException
,MyEqualRelationException
,MyRelationNotFoundException
,MyEqualGroupIdException
,MyEqualMessageIdException
,MyGroupIdNotFoundException
,MyPersonIdNotFoundException
。其中MyEqualPersonIdException
继承了EqualPersonIdException
,MyEqualRelationException
继承了EqualRelationException
,MyPersonIdNotFoundException
继承了PersonIdNotFoundException
,MyRelationNotFoundException
继承了RelationNotFoundException
,MyEqualGroupIdException
继承了EqualGroupIdException
,MyEqualMessageIdException
继承了EqualMessageIdException
,MyGroupIdNotFoundException
继承了GroupIdNotFoundException
,MyPersonIdNotFoundException
继承了PersonIdNotFoundException
。
在普通类中MyPerson
与前一次作业大致相同。其中有三个容器HashMap<Integer, Person> acquaintance
,HashMap<Integer, Integer> value
,ArrayList<Message> messages
,用来存储熟人队列和熟人之间的value值和对应的消息对象。本次作业的只对MyPerson添加了存储消息对象的功能,其他部分与上一次作业相同。
在MyGroup
中使用HashMap
来记录组内的所有的人,HashMap<Integer, Person> people
。MyGroup
类是上一次作业中不包含的类,在本次作业中含义是一个组,组中可以存储Person
的对象。
在MyNetwork
中拥有4个容器,HashMap<Integer, Person> people
,HashMap<Integer, Group> grouphash
,HashMap<Integer, Message> meshash
,HashMap<Integer, Integer> reps
。people
用来存储社交网络里的所有人,grouphash
用来存储社交网络里面的所有的组,meshash
用来存储社交网络里的所有的消息,reps
则是并查集时候用来记录代表元的哈希字典。与上一次作业相比,增加了记录grouphash
的哈希字典,和增加了记录消息对象Message
的哈希字典。
性能问题
第三单元正确性还是很容易实现的,因为每一个方法都有很多的jml语句来解释,近似伪代码的jml语句可以稍作修改成为java程序的一部分。
但是这样的实现很容易造成程序的时间复杂度过高。在本次作业中,一切的O(n^2)
的方法都可能会ctle。本次作业中最容易超时的方法是queryBlockSum
,isCircle
和getValueSum
。 isCircle
,queryBlockSum
如果用宽度优先搜索的话,时间复杂度是O(E+V)
,这样的时间复杂度在第一次作业中不会出现超时的问题,但是在本次作业中指令条数可以是10000条,cpu限制仍然是2s,这样实现的话将会在本次测评中遇到ctle的问题。因此,在bug修复阶段,我将宽度优先搜索转换成为并查集算法,以均摊O(1)
的时间复杂度完成连通分量的统计和两点连通性的判断。
getValueSum
如果按照jml直接翻译的话会有o(n^3) or o(n^2)
的时间复杂度,这样的时间复杂度是肯定会超时的。getValueSum
如果简简单单地按照jml书写的方式使用二重遍历的话肯定会超时。因此,在bug修复阶段,我使用缓存的思想来牺牲空间换取时间。在每一次addPerson
或者delPerson
或者addRelation
的时候都需要对缓存的量进行修改。经过这样的修改,qgvs
指令的执行将会变成均摊O(n)
的时间复杂度,可以满足程序运行时间的限制。
此外,容器的选择也是十分重要的问题。如果涉及到查找的容器应尽可能选择哈希类型的容器,如:HashMap<Integer, Person> people
,HashMap<Integer, Group> grouphash
,HashMap<Integer, Message> meshash
,HashMap<Integer, Integer> reps
,HashMap<Integer, Person> acquaintance
,HashMap<Integer, Integer> value
。容器选择ArrayList
来存储所有的数据的话,也会产生查询时间复杂度过大的问题,有可能造成cpu超时。对比HashMap
查找的O(1)的时间复杂度,Arraylist查询复杂度是O(n)
,在我们的程序中大部分方法都会去查询容器,降低查询的时间复杂度是拒绝ctle的重要手段之一。
MyBUG&&测评与互测
测评
通过中测,强测挂,互测挂
在提交作业之前要进行大量有效的测试,尽可能穷举所有的边界情况。同时可以采用自动测评机的方式对代码进行功能性的测试。当然,设计测评机的过程也是对于题目要求的再一次明确。
本次作业我虽然在课下有进行测试,但是自己生成的测试数据不够优秀,没有主动去构造容易超时的数据,即使构造了运行时间会长一点的数据也没有具有很好的数据质量。本地测试的结果是完全不会ctle,那么我就满怀信心地交到了平台上……
理想很美好,但是现实很残酷。强测和互测的结果给了我当头一棒,除了超时还是超时,我一下子无所适从,在时间复杂度的泥沼中反复挣扎,险些丧命。
在本次作业中addToGroup的1111数据限制我也没有注意到。虽然助教有在讨论区里面修改了此部分的jml规格,但是慵懒的我完全没有注意到这样的修复。程序就这样想当然地出错了。此bug为程序的正确性出错,实属不应该,之后一定要吸取本次的经验教训,多读jml,多测试!
互测
当在互测屋子里面发现hack别人是如此容易的时候,我就意识到本次作业可能真的是崩了,仅根据程序的正确性就能hack同屋子里面的很多人,自己深陷妄自菲薄的状态煎熬地等待强测结果的发布。强测结果可想而知,在悲伤之余也认真反思自己的不足。虽然自己有搭建测评机测试自己的程序,但是没有生成能够让自己超时的数据在提交之前更正自己的错误。1111数据限制没有认真关注jml的更新,想当然地对程序进行实现,最后程序的正确性出现了较大的问题。
在本次作业中我被找出bug(两个针对qgvs,一个针对qbs),也找出别的同学的bug(正确性bug,qgvs时间复杂度过高)。屋子里面的同学大多犯的错误和我类似,要么是时间复杂度过高,要么是jml语言没有好好阅读。大家在屋子里面砍得不亦乐乎,“己所不欲勿施于人”是不可能的。完全不需要看别人的代码,构造一个能把自己hack的数据,大概率屋子里面的其他人也就会被hack了。
心得体会
在本次作业中,我的信心被击溃,整个人在数不清的bug中苦苦挣扎。时间复杂度成为测评中最致命的错误,在同学们的“千刀万剐”下我清楚地认识到了在本次作业中时间复杂度的重要性。后知后觉的我只能被各种数据蹂躏,却毫无还手之力。
因为错误太多,bug修复都变得异常困难。修复了此处,另外一处的数据就超时了。修复了那处,原本不超时的数据就又超时了。我就这样在无底的沼泽中苦苦挣扎,哀求老天放过我吧,下一次我肯定不这么写时间复杂度这么高的代码了,让我修复成功吧,我保证下一次作业重新做人,好吗?
本次作业体验极差,我为我自己的懒惰、愚笨付出了惨痛的代价,自己在bug修复阶段一次又一次强行扶起“扶不起的代码”。我完全没有任何兴致再去多读一遍本单元第二次作业的代码,一看到第十次作业的全是bug的代码就头疼。同时对助教们有耐心去看完我的混乱代码修复表示由衷的感激与敬佩。
一个程序的质量是由整体架构决定的,有一些程序就是没有修复的必要,因为修复意味着重构。本次作业的代码真的没有什么修改的价值了,要想让此程序能够恢复正常也许就只能依靠大换血了。本次作业的程序恶心得令人发指,让我实在不愿意多看他一眼。罢了,下次作业一定要洗心革面。
第三次作业
问题概要
实现一个社交关系模拟系统,通过实现官方提供的接口 Person
、Network
和 Group
,来实现自己的 Person
、Network
和Group
类。通过实现官方提供的接口 Message
、EmojiMessage
、NoticeMessage
和 RedEnvelopeMessage
,来实现自己的 Message
、EmojiMessage
、NoticeMessage
和 RedEnvelopeMessage
类。构造EqualGroupIdException
,EqualMessageIdException
,EqualPersonIdException
,EqualRelationException
,GroupIdNotFoundException
,MessageIdNotFoundException
,PersonIdNotFoundException
,RelationNotFoundException
,EmojiIdNotFoundException
,EqualEmojiIdException
异常类的子类来实现程序中的异常处理。
设计策略和容器选择
本次作业是jml单元的第三次作业,我的实现方式是完全照搬已有的jml规格进行书写。jml中已经明确地指出了数据规格和方法规格。在理解这些规格之后我将这些规格使用java语言表达,编写出符合jml语义限制的程序。
在本单元的第三次作业中,我是将所有的代码编写放在了阅读jml之后。在阅读完全部的jml后,我对整个程序的架构有了一定的了解。在完成本次作业的最初阶段,首先选择合适的容器来满足jml的数据规格,之后按照方法规格完成每一个要填充的方法。
在严格的jml约束下,选择合适的容器是本次作业最重要的程序设计环节。合理的容器选择能够为程序构建良好的架构,降低程序运行的时间复杂度,有利于提升程序的可读性和正确性,降低程序运行出错的可能。
在本次的设计中,在普通类中MyPerson
中有三个容器HashMap<Integer, Person> acquaintance
,HashMap<Integer, Integer> value
,ArrayList<Message> messages
,用来存储熟人队列和熟人之间的value值和对应的消息对象。在MyNetwork中拥有6个容器,HashMap<Integer, Person> people
,HashMap<Integer, Group> grouphash
,HashMap<Integer, Message> meshash
,HashMap<Integer, Integer> reps
,HashMap<Integer, Integer> emojitimeHash
,HashMap<Integer, HashSet<Integer>> emojiIdToId
。people
用来存储社交网络里的所有人,grouphash
用来存储社交网络里面的所有的组,meshash
用来存储社交网络里的所有的消息,reps
则是并查集时候用来记录代表元的哈希字典,emojitimeHash
记录的是以emojiid
为key
,次数为value
的键值对,emojiIdToId
是一个二维容器,用来根据一个emojiid
找到所有相同emojiid
的id
。
同第一次作业一样,因为容器内部的元素无需有序,并且对容器中元素的查找比较频繁,所以大部分的容器选择HashMap
容器来存储数据。
完成了容器的合理选择,那么整个程序的编写变得十分简单,按照jml的方法规格逐一完成每一个方法即可。
UML类图
作业架构
通过UML类图我们可以看到作业的整体架构就是普通类+异常类。普通类包含MyPerson
,MyNetwork
,MyMessage
,MyGroup
,MyEmojiMessage
,MyNoticeMessage
,MyRedEnvelopeMessage
,其中MyPerson
实现了Person
接口,MyNetwork
实现了Network
接口,MyMessage
实现了Message
接口,MyGroup
实现了Group
接口,MyEmojiMessage
实现了EmojiMessage
接口继承了MyMessage
类,MyNoticeMessage
实现了NoticeMessage
继承了MyMessage
类,MyRedEnvelopeMessage
实现了RedEnvelopeMessage
继承了MyMessage
类。
异常类包含MyEqualPersonIdException
,MyEqualRelationException
,MyRelationNotFoundException
,MyEqualGroupIdException
,MyEqualMessageIdException
,MyGroupIdNotFoundException
,MyPersonIdNotFoundException
,MyEmojiIdNotFoundException
,MyEqualEmojiIdException
,MyMessageIdNotFoundException
。MyEqualPersonIdException
继承了EqualPersonIdException
,MyEqualRelationException
继承了EqualRelationException
,MyPersonIdNotFoundException
继承了PersonIdNotFoundException
,MyRelationNotFoundException
继承了RelationNotFoundException
,MyEqualGroupIdException
继承了EqualGroupIdException
,MyEqualMessageIdException
继承了EqualMessageIdException
,MyGroupIdNotFoundException
继承了GroupIdNotFoundException
,MyPersonIdNotFoundException
继承了PersonIdNotFoundException
,MyEmojiIdNotFoundException
继承了EmojiIdNotFoundException
,MyEqualEmojiIdException
继承了EqualEmojiIdException
,MyMessageIdNotFoundException
继承了MessageIdNotFoundException
。
在本次的设计中,在普通类中MyPerson
中有三个容器HashMap<Integer, Person> acquaintance
,HashMap<Integer, Integer> value
,ArrayList<Message> messages
,用来存储熟人队列和熟人之间的value
值和对应的消息对象。在MyNetwork
中拥有6个容器,HashMap<Integer, Person> people
,HashMap<Integer, Group> grouphash
,HashMap<Integer, Message> meshash
,HashMap<Integer, Integer> reps
,HashMap<Integer, Integer> emojitimeHash
,HashMap<Integer, HashSet<Integer>> emojiIdToId
。people
用来存储社交网络里的所有人,grouphash
用来存储社交网络里面的所有的组,meshash
用来存储社交网络里的所有的消息,reps
则是并查集时候用来记录代表元的哈希字典,emojitimeHash
记录的是以emojiid
为key
,次数为value
的键值对,emojiIdToId
是一个二维容器,用来根据一个emojiid
找到所有相同emojiid
的id
。
同第一次作业一样,因为容器内部的元素无需有序,并且对容器中元素的查找比较频繁,所以大部分的容器选择HashMap
容器来存储数据。
性能问题
本次作业的正确性实现相较于前两次作业有了难度上较大的提升。sendIndirectMessage,deleteColdEmoji
的jml方法约束冗长,同学们在阅读的时候很容易迷失自我,错误理解jml想要表达的原意。此外,本次作业中引入了继承关系下的jml语言实现的表述,合理理解不同类之间的继承关系也是本次作业中十分重要一个环节。
在本次的时间复杂度实现上,第三次作业没有提出太高的要求sendIndirectMessage
,deleteColdEmoji
是两个可能会产生超时的方法。在sendIndirectMessage
中需要寻找两点之间的最短路径,采用堆优化的迪杰斯特拉算法可以让此方法的时间复杂度达到O(nlog(n))
,因此不会超时。
此外,容器的选择也是十分重要的问题。如果涉及到查找的容器应尽可能选择哈希类型的容器,如:HashMap<Integer, Person> people
,HashMap<Integer, Group> grouphash
,HashMap<Integer, Message> meshash
,HashMap<Integer, Integer> reps
,HashMap<Integer, Person> acquaintance
,HashMap<Integer, Integer> value
。容器选择ArrayList
来存储所有的数据的话,也会产生查询时间复杂度过大的问题,有可能造成cpu超时。对比HashMap
查找的O(1)
的时间复杂度,Arraylist
查询复杂度是O(n)
,在我们的程序中大部分方法都会去查询容器,降低查询的时间复杂度是拒绝ctle的重要手段之一。
MyBug&&测试与互测
通过中测,通过强测,通过互测
互测
通过自动测评机检测同学们的代码,什么都没发现,可能是自己生成的数据还是不够强?
屋子里面一次hack都没有成功,同学们的代码大致没有问题。
基于JML规格来设计测试的方法和策略
-
OpenJml
可以对jml注释进行正确性的检查,同时可以进行静态检查和动态断言检查
-
JMLUnitNG
可以根据规格自动化生成测试样例,进行单元测试
-
Junit
自己编写测试代码,可以对覆盖率进行检查
通过使用以上的测试插件、测试工具进行测试,发现这些工具在这三次作业中的用处不大,繁琐且测试效果不够好。
效果并没有比我写的垃圾测评机好。
心得体会
OO第三单元外表单纯,但是陷阱很多,稍不注意时间复杂度,就能收获强测和互测的双双爆炸。
本单元第一次作业在完成时甚是小心,首先是自己在之前没有接触过JML
语法,因此自己也多去参考互联网上的学长学姐的博客来学习理解JML
。在isCircle
方法中也按照博客中推荐的宽度优先搜索来实现自己的代码。在本次作业中也没有在强测和互测中遇到超时的问题。
第一次作业难度较小,程序的正确性容易保证,强测没有太针对程序的运行时间单独构造特殊样例。让我产生了一种第三单元十分简单的错觉。在第一次的代码实现中,自己使用了宽度优先搜索作为queryBlockSum
和isCircle
的实现方式,在本次作业中因为程序最多只有1000条,所以采用宽度优先搜索的算法并不会产生超时的错误。
本单元第二次作业我在完成时变得佛系。以为按照上一次作业的架构进行填充就好,isCircle
和queryBlockSum
也没有对上一次使用的宽度优先搜索进行修改。新添加的方法也几乎是按照jml
语法中生搬硬套到了我的代码中。就这样自己的代码在强测和互测中多次爆炸。qbs
和qgvs
时间复杂度过高,手动构造数据一测一个准。同时,因为自己没有好好关注课程组的jml更新,1111人数限制没有添加,导致程序的正确性出现大问题,强测就有两个针对1111容器限制的点错误。
当在互测屋子里面发现hack别人是如此容易的时候,我就意识到本次作业可能真的是崩了,仅根据程序的正确性就能hack同屋子里面的很多人,自己深陷妄自菲薄的状态煎熬地等待强测结果的发布。强测结果可想而知,在悲伤之余也认真反思自己的不足。虽然自己有搭建测评机测试自己的程序,但是没有生成能够让自己超时的数据在提交之前更正自己的错误。1111数据限制没有认真关注jml的更新,想当然地对程序进行实现,最后程序的正确性出现了较大的问题。
本次的作业给了我极大的教训,不要想当然,要关注课程组的更新,要反复阅读jml,要对自己的每一个方法进行时间复杂度的分析。不要轻视面向对象的第三单元,要从学长学姐的博客中获取关于超时经验……
在经历了第二次作业的打击之后,书写第三次作业的我战战兢兢,如履薄冰。生怕自己的程序再一次超时,反复校验jml语法并且对于时间复杂的方法进行多次优化,自己也在极度不安中提交了代码,等待课程组对我的审判……
还好,没超时,自己也结束了担忧与恐惧。
感觉在第三单元收获蛮大,之前的自己写程序对于时间复杂度毫不担心,大多数时候的程序都是冗长并且运行极慢。OO第三单元让我第一次认识到了时间复杂度在程序设计的时候是如此重要。没有实现合理的时间复杂度的话程序一定会超时,强测一定会挂掉。在事后反思自己第二次作业中出现的种种问题,感觉自己所作所为真的是又可气又好笑。在明明知道有并查集算法的时候为什么不去替换已有的宽度优先搜索的算法呢?为什么会如此自信地认为因为自己已经测试过时间,所以强测不会超时?总的来说,OO就是要我们拒绝想当然,要耐心,要细心。希望自己能永远记住本单元第二次作业的惨痛经历,不要让类似的事情重演。
本单元的三次作业和前两个单元有较大差距。本单元的三次作业主要是契约式编程。在接连的三次作业中,同学们对于jml的理解逐渐加深。本单元三次作业的内容也没有包含自己书写jml语言的部分,这使得本单元的难度得到很好的控制。
经过本单元的三次训练,我也学习了很多图论的算法。Dijkstra算法,宽度优先搜索算法,并查集算法……