OO第三单元个人总结

JML理论与基础与应用工具链

JML是什么?

  • Java建模语言(JML)是一种行为接口规范语言,可用于指定Java模块的行为 。它结合了Eiffel的契约设计方法 和Larch 系列接口规范语言的基于模型的规范方法 ,以及细化演算的一些元素 。草稿 《Design by Contract with JML》(由Gary T.Leavens和Yoonsik Cheon撰写)解释了JML作为Java的契约式设计(DBC)语言的最基本用法。

JML基础语法

  • 原子表达式

    \result:表示此方法执行后的返回值。

    \old(expr):表示一个表达式expr在方法执行之前进行计算后的取值。

  • 量词表达式

    \forall:全称量词,对于每个在范围内的变量,若全部满足表达式,则返回真。

    \exists:特称量词,对于每个在范围内的变量,若存在一个变量满足表达式,则返回真。

    \sum:返回给定范围内的表达式的和。

  • 前置条件

    require P;使用者必须满足前置条件,即P为真。

  • 后置条件

    ensures P;实现者依据前置条件,保证实现过程满足P的要求。

  • 副作用

    assignable(可赋值)、modifiable(可修改)

    /nothing关键词表示没有任何可修改的变量,/everything关键词表示可修改任何变量

  • 异常处理

    signals (***Exception e) b_expr当b_expr为真时,方法抛出异常e。

    signals_only (***Exception e)满足前置条件即抛出异常e。

  • 不变式

    invariant P;在所有可见状态下都要求满足P为真。

  • 状态变化约束

    constraint P;前序可见状态和当前可见状态的约束。

JML工具链

  • openjml:对JML注释的完整性进行检查。检查包括经典的类型检查、变量可见性与可写性等。通过命令行使用OpenJML时,可以通过-check参数(缺省)指定类型检查。

  • SMT Solver:一般为检查代码规格,生成运行时测试等。与openjml一同使用。

  • JMLUnitNG:针对类自动生成测试样例并进行测试。

部署SMT Solver

  • /*@ public normal_behavior
    @ ensures \result == a1 + a2;
    @*/
    public int sum(int a1, int a2) {
    return a1 + a2;
    }

  • /*@ public instance model int[] arr;
    @*/
    private ArrayList<Integer> array = new ArrayList<>(); /*@ public normal_behavior
    @ requires arr != null;
    @ ensures (\forall int i,j; 0 <= i && i <= j && j < arr.length; arr[i] <= arr[j]);
    @ also
    @ public exceptional_behavior
    @ signals (NullPointerException e) arr == null;
    @*/
    public void sorted() {
    Collections.sort(array);
    }

  • 对第11次作业进行检查

部署JMLUnitNG/JMLUnit

[TestNG] Running:
Command line suite Passed: racEnabled()
Passed: constructor MyGroup(-2147483648)
Passed: constructor MyGroup(0)
Passed: constructor MyGroup(2147483647)
Failed: <<MyGroup@800003e1>>.addPerson(null)
Failed: <<MyGroup@800003c2>>.addPerson(null)
Failed: <<MyGroup@3e1>>.addPerson(null)
Failed: <<MyGroup@800003e1>>.delPerson(null)
Failed: <<MyGroup@800003c2>>.delPerson(null)
Failed: <<MyGroup@3e1>>.delPerson(null)
Passed:<<MyGroup@800003e1>>.equals(null)
Passed:<<MyGroup@800003c2>>.equals(null)
Passed:<<MyGroup@3e1>>.equals(null)
Passed: <<MyGroup@800003e1>>.equals(java.lang.Object@800003e1)
Passed: <<MyGroup@800003c2>>.equals(java.lang.Object@800003c2)
Passed: <<MyGroup@3e1>>.equals(java.lang.Object@3e1)
Passed: <<MyGroup@800003e1>>.getAgeMean()
Passed: <<MyGroup@800003c2>>.getAgeMean()
Passed: <<MyGroup@3e1>>.getAgeMean()
Passed: <<MyGroup@800003e1>>.getAgeVar()
Passed: <<MyGroup@800003c2>>.getAgeVar()
Passed: <<MyGroup@3e1>>.getAgeVar()
Passed: <<MyGroup@800003e1>>.getConflictSum()
Passed: <<MyGroup@800003c2>>.getConflictSum()
Passed: <<MyGroup@3e1>>.getConflictSum()
Passed: <<MyGroup@800003e1>>.getId()
Passed: <<MyGroup@800003c2>>.getId()
Passed: <<MyGroup@3e1>>.getId()
Passed: <<MyGroup@800003e1>>.getRelationSum()
Passed: <<MyGroup@800003c2>>.getRelationSum()
Passed: <<MyGroup@3e1>>.getRelationSum()
Passed: <<MyGroup@800003e1>>.getValueSum()
Passed: <<MyGroup@800003c2>>.getValueSum()
Passed: <<MyGroup@3e1>>.getValueSum()
Failed: <<MyGroup@800003e1>>.hasPerson(null)
Failed: <<MyGroup@800003c2>>.hasPerson(null)
Failed: <<MyGroup@3e1>>.hasPerson(null)
Failed: <<MyGroup@800003e1>>.updateRelation(null)
Failed: <<MyGroup@800003c2>>.updateRelation(null)
Failed: <<MyGroup@3e1>>.updateRelation(null)

如上,得到对MyGroup测试结果,基本为边界条件,如int最大最小和零,以及引用为null的情况。

作业架构分析

第九次作业

本次作业要求通过Person、Network接口和其中的JML规范,实现自己的MyPerson、MyNetwork类,总体难度不大,依据JML完成实现即可。

由于性能的要求,我们需要寻找合适的容器实现交互网络中的Person的存储和每个Person对应的联系人Person的存储,在这里我使用HashMap作为容器,由于可能需要大量的contians判断一个Person对象是否已经存储在容器中,因此使用Hashmap帮助我们完成散列表的查找,以常数的时间复杂度完成。

对于isCircle的判断,我采用BFS广度搜索来判断图的连通性,由于使用了图论算法,我构造了MyMap对顶点进行管理,并实现了MyNode管理一个点内的数据结构。

第十次作业

本次作业增加了Group接口和JML规范来实现对群组中的人的模拟,增加了RelationSum、ValueSum、ConflictSum、AgeMean、AgeVar等方法,这些方法的普通实现都需要对群组中的容器进行遍历,性能损失极大,因此这里采用了对RelationSum、ValueSum、ConflictSum的缓存,在对Group中增加Person时,更新相应的数值,从而避免一次又一次的重复遍历。

由于AgeVar在计算方差时,使用一些特殊的公式会造成精度不足的情况,因此在实现中仅采用遍历计算。

第十一次作业

本次作业增加了borrowFrom、queryMoney等与对应的Person中money有关的方法,实现并不困难,重点在于queryStrongLinked、queryMinPath以及queryBlockSum方法,对性能和算法都有着较高的要求。

queryStrongLinked没有使用tarjan等困难的算法,由点的双连通的特性,通过寻找出一条连通路径,除去两头端点,遍历剩余各点,每次只删除遍历到的一个点,然后进行bfs运算,观察是否连通,运用这种暴力方法,可以检查这个路径中是否存在割点,也就是说,两个点之间是否存在割点,若存在,则必不可能存在点的双连通性,也就计算出了结果。

queryMinPath方法的实现采用了Dijsktra算法加利用TreeSet的最小值计算优化,由于TreeSet本身采用红黑树实现,在我们寻找最小权重的点时,极大地简化了时间复杂度。

queryBlockSum使用了并查集的算法,在每次addRelation时,将两个点的祖先寻找到,并将其中较长的分支,插入较短的分支中,保持其不退化。最后,遍历割点,计算没有祖先,即祖先是自己的点的数量,即可得到Block的数量。

Bug与修复

第九次作业

本次作业在在强测中没有遇到问题。

在互测中,主要构造边界数据,尽量制造可能TLE的数据,同时。利用对拍器和随机测试数据生成器完成正确性检验,在此次互测中没有hack到别人。

第十次作业

本次作业由于在五一期间,当时正在外旅游,在最初并未实现缓存,仅仅使用ArrayList加快遍历速度,并且根据对称性减半了遍历的次数,但由于时间紧张,没有进行全面的测试,未发现双重for循环中i、j终止条件写错了,导致强测爆零,甚至没有进入互测。

在修复bug中,除了上述bug,增加了缓冲机制,解决了TLE的问题。

第十一次作业

本次作业强测让人及其糟心,四个点t了,但重新提交一次,就能过一个点,另外两个点被生生卡在超过时限0.0几秒的地方。在研讨课中有同学提到初始容量问题,遂减小初始大小,过了……(说好边界不卡的)

互测中同样利用自动化评测机对其他代码疯狂输出,可惜未发现问题,在查看其代码与JML匹配程度中也未发现问题。

心得体会

JML是一种面向规格的设计语言,他让我们依据规格,能够正确的使用方法和数据,能够写出完善的功能实现,并且在保证正确性的条件下,选择不同的实现方式,这种代码编写方式让人十分舒服,写的有理有据。

JML代码规格,说白了又是一次离散数学的学习,通过形式化语言的表述,让程序员更加精准的完成代码,也让测试变得更加切中肯綮,总体来说,为我们的学习之路打开了一扇新的窗户。

OO第三单元个人总结的更多相关文章

  1. OO第三单元作业总结

    OO第三单元作业总结--JML 第三单元的主题是JML规格的学习,其中的三次作业也是围绕JML规格的实现所展开的(虽然感觉作业中最难的还是如何正确适用数据结构以及如何正确地对于时间复杂度进行优化). ...

  2. 规格化设计——OO第三单元总结

    规格化设计--OO第三单元总结 一.JML语言理论基础.应用工具链 1.1 JML语言 ​ JML(java modeling language)是一种描述代码行为的语言,包括前置条件.副作用等等.J ...

  3. 【OO学习】OO第三单元作业总结

    [OO学习]OO第三单元作业总结 第三单元,我们学习了JML语言,用来进行形式化设计.本单元包括三次作业,通过给定的JML来实行了一个对路径的管理系统,最后完成了一个地铁系统,来管理不同的线路,求得关 ...

  4. OO第三单元(地铁,JML)单元总结

    OO第三单元(地铁,JML)单元总结 这是我们OO课程的第二个单元,这个单元的主要目的是让我们熟悉并了解JML来是我们具有规格化编程架构的思想.这个单元的主题一开始并不明了,从第一次作业的路径到第二次 ...

  5. OO第三单元——基于JML的社交网络总结

    OO第三单元--基于JML的社交网络总结 一.JML知识梳理 1)JML的语言基础以及基本语法 JML是用于java程序进行规格化设计的一种表示语言,是一种行为接口规格语言.其为严格的程序设计提供了一 ...

  6. OO第三单元作业——魔教规格

    OO第三单元作业--魔教规格 JML的理论基础和相关工具   JML(Java Modeling Language,Java建模语言),在Java代码种增加了一些符号,这些符号用来标志一个方法是干什么 ...

  7. 2020 OO 第三单元总结 JML语言

    title: 2020 OO 第三单元总结 date: 2020-05-21 10:10:06 tags: OO categories: 学习 第三单元终于结束了,这是我目前为止最惨的一单元,第十次作 ...

  8. 2020北航OO第三单元总结

    2020北航OO第三单元总结 本单元要求是根据JML规格完善代码,初看是一个简单的代码照搬实现的东西,但最后才发现由于CPU时间的限制,还考察了大量优化策略及数据结构中关于图的知识,是一次非常注重细节 ...

  9. OO第三单元作业(JML)总结

    OO第三单元作业(JML)总结 目录 OO第三单元作业(JML)总结 JML语言知识梳理 使用jml的目的 jml注释结构 jml表达式 方法规格 类型规格 SMT Solver 部署JMLUnitN ...

随机推荐

  1. js 创建简单的表单同步验证器

    SyncValidate declare const uni: any; export interface SyncValidateOpt { [key: string]: SyncValidateF ...

  2. 高倍币VAST了解一下,如何掀起算力挖矿新热潮?

    随着比特币.以太坊等主流数字货币的起起落落,市场对于数字货币交易似乎进入了冷却期.很多生态建设者开启了观望态度,机构以及巨鲸们也开始纷纷着手分散投资.就在此时,一个新的概念逐步露出头角,吸引了大众关注 ...

  3. IDEA 敏捷开发技巧——后缀完成

    前言 "工欲善其事,必先利其器." 所以说今天来看一看如何压榨 IDEA ,让你的 IDEA 使用的更顺手! 今日技巧: 后缀完成 自定义后缀完成模版 示例 上面动图使用了 .so ...

  4. pytorch中修改后的模型如何加载预训练模型

    问题描述 简单来说,比如你要加载一个vgg16模型,但是你自己需要的网络结构并不是原本的vgg16网络,可能你删掉某些层,可能你改掉某些层,这时你去加载预训练模型,就会报错,错误原因就是你的模型和原本 ...

  5. (转)linux下的系统调用函数到内核函数的追踪

    转载网址:http://blog.csdn.net/maochengtao/article/details/23598433 使用的 glibc : glibc-2.17使用的 linux kerne ...

  6. [JAVA学习笔记]JAVA基本程序设计结构

    一个简单的Java应用程序 public class FirstSample { public static void main(String[] args) { System.out.println ...

  7. 移动端时间回显iphone出现的问题

    new Date(item.startTime.replace(/-/g, '/') dateFormat('hh:mm', new Date(item.startTime.replace(/-/g, ...

  8. SpringBoot 整合 hibernate 连接 Mysql 数据库

    前一篇搭建了一个简易的 SpringBoot Web 项目,最重要的一步连接数据库执行增删改查命令! 经过了一天的摸爬滚打,终于成功返回数据! 因为原来项目使用的 SpringMVC + Hibern ...

  9. DOM及相关操作

    1.背景介绍        什么是DOM?简单地说,DOM是一套对文档的内容进行抽象和概念化的方法, 在现实世界里,人们对所谓的'世界对象模型'都不会陌生,例如,当用'汽车'.'房子'和'树'等名词来 ...

  10. vue子组件的样式没有加scoped属性会影响父组件的样式

    scoped是一个vue的指令,用来控制组件的样式生效区域,加上scoped,样式只在当前组件内生效,不加scoped,这个节点下的样式会全局生效. 需要注意的是:一个组件的样式肯定是用来美化自己组件 ...