哈工大软件构造Lab2导读 - Stanford 6.031 Problem Set 2: Poetic Walks

本文主要是介绍哈工大软件构造Lab2导读 - Stanford 6.031 Problem Set 2: Poetic Walks,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 前言
  • 一、P1-Problem 1:编写测试用例
    • 1.GraphStaticTest
    • 2.GraphInstanceTest
  • 二、实现两个ADT
    • 1.AF,RI,Safety from Rep exposure .etc
    • 2.checkRep ( )
    • 3.方法的具体实现
    • 4.实现Graph.empty ( )
  • 三、Poetic Walks
    • 1.题目意思梳理(结合MIT页面和spec看)
    • 2.编写测试用例
    • 3.具体实现Poet
      • 1) 构造方法GraphPoet
      • 2) 生成句子方法Poem
      • 3) toString方法
    • 4.测试代码覆盖度
  • 四、重构Lab 1中的Social Network
  • 总结


前言

由于实验的介绍为MIT的全英文页面,要点分散,可能造成理解上的困难。所以在这里梳理总结一下软构lab2的整个流程,可以结合MIT的问题描述来看。希望能帮到后来的学弟学妹,不要拿到程序包直接两眼一抹黑,不知从何下手。

写下这篇文章是在完成实验后进行回忆,如有疏漏请多指正。

先简单总结Lab 2完成的工作:从github将代码clone到本地,根据已有spec完成测试用例的编写(P1,Problem 1),再分别编写两个ADT: ConcreteVerticesGraphConcreteEdgesGraph的具体实现(P1,Problem 2 / 3)。之后,我们利用两个ADT分别实现具体的类GraphPoet,并完成一系列的工作(P1,Problem 4)。并重构我们在Lab 1中实现过的人际关系图(P2)

本实验的完成基于IDEA,可以方便地可视化代码覆盖度,所以没有涉及安装EclEmma。


一、P1-Problem 1:编写测试用例

按照TDD(Test-Driven Development,测试驱动开发)的策略,在编写具体实现之前,我们需要根据已有的spec设计出对应方法的测试。

1.GraphStaticTest

这一部分是对Graph.empty()的测试,可以先不用管,在完成 Problem 3.2 实现Graph.empty()之前,这一块的代码是跑不起来的XD
在这里插入图片描述
对GraphStaticTest的补充也需要在Problem 3.2中进行。

2.GraphInstanceTest

这一部分是对Graph中的具体方法进行测试,需要好好构思构思,如果刚开始草草写了/没有考虑周全,那么在后面代码覆盖度过低,会反复回头来修改这部分,耽误很多时间!

具体而言,这一部分要按等价类划分的方法,编写测试用例。严格写下来的话,代码行数不少。举个栗子:

Testing strategy
testRemove( )
测试remove( )方法,等价类划分如下:
删除点是否在图中:是,否
删除点是否有相连边:是,否

排除不可能的情况,共3个等价类,所以需要写三组测试。

另外,提醒考虑ADT对有环图的支持并测试,后续部分会用到。

二、实现两个ADT

1.AF,RI,Safety from Rep exposure .etc

具体实现两个ADT,首先要求我们撰写相信很多同学这是第一次实际写一个程序的AF、RI,这些概念的具体意义可以参见 关于AF, RI, Rep exposure
其实举个例子,照葫芦画瓢也还算容易。例如,对于ConcreteEdgesGraph,它的这些概念可以解释如下:

Abstraction function:
AF(vertices) = Graph中的点
AF(edges) = Graph中的边
 
Representation invariant:
点的名字不能重复
所有点都在vertices中
edge的权值为正
两点间的单向边最多只能有一条
 
Safety from rep exposure:
成员变量vertices与edges均用private final修饰,防止其被外部修改
在涉及返回内部变量时,采用防御性拷贝的方式,创造一份新的变量return

2.checkRep ( )

checkRep ( )部分就是针对RI设计出检查代码。比如,在RI中有一条“edge的权值为正”,则在checkRep ( )中需要用assert语句检查edge的成员是否大于0,在每次运行方法(如add,set)且修改了成员变量后,需要在return前调用checkRep ( )进行检查

3.方法的具体实现

在实现ConcreteVerticesGraphConcreteEdgesGraph时,MIT的页面要求先用String型,再用泛型L,其实可以直接用泛型L来写,节省后续改的工作量。若用泛型L来写,则需要将两个ADT的部分内容修改为如下:

public class ConcreteEdgesGraph<L> implements Graph<L> { ... }class Edge<L> { ... }

public class ConcreteVerticesGraph<L> implements Graph<L> { ... }class Vertex<L> { ... }

相关方法的具体实现在这里暂且不表,在完成方法的实现后,可以通过前面写的测试用例来检验方法编写是否正确,并看看代码覆盖度,有哪些代码没有覆盖到。(实验要求代码覆盖度尽可能达到100%

在这里简单讲一下IDEA的代码覆盖度测试:
在这里插入图片描述
在这里插入图片描述
运行之后:
在这里插入图片描述

在这里插入图片描述
另外注意,两个ADT中需要分别实现成员Edge与成员Vertex,在ConcreteEdgesGraphTestConcreteVerticesGraphTest中,需要完成对EdgeVertex各自成员方法的测试。

4.实现Graph.empty ( )

这一部分要求我们对Graph.java中的empty ( )方法进行实现,具体来说,就是让empty ( )方法返回两个具体ADT中的一个,如图:
在这里插入图片描述
完成后,在GraphStaticTest中补充测试代码,具体来说,就是模仿已有代码,验证在Interger,Double等类型下是否正确,如图:
在这里插入图片描述

三、Poetic Walks

到此,我们已经完成了MIT页面上的Problem 1~3,开始解决Problem 4。

1.题目意思梳理(结合MIT页面和spec看)

在这一步中,需要依据语料生成一个单词图,单词图的每个顶点是语料中的一个单词,单词图的边代表前一个单词紧接着后一个单词,边的权重为前一个单词紧接着后一个单词的次数,并且不考虑大小写与标点符号。举个例子:

hello,hello,HeLlo,world!

其中,hello→hello出现两次,hello→world出现一次,则这个语料构成的图为:
在这里插入图片描述
更加具体的示例可以参照MIT实验官网的页面。

在根据语料构造好图后,下面需要完成的任务是:输入一个句子,提取出句子两两相邻的单词对,在图中进行一次检索,若句子的前后两单词w1→w2在单词图中隔着一个顶点b,则将b加入句子中,得到w1→b→w2。具体的构建规则如下:

① w1与w2间只能间隔一个顶点,这也就是说,如果在图中出现了w1→a→b→w2,甚至间隔更多的情况,则不会在w1→w2间加入单词。

② 如果从w1到w2同时有两条路径w1→a→w2与w1→b→w2,则选择权重最高路径上的单词加入w1与w2之间。

③ 输出的句子中原单词的大小写保持不变,加入的单词全用小写。

④ 可以存在指向自己的边,即该图可以是带环图。

2.编写测试用例

在理清思路后,我们可以着手开始编写测试用例,具体的测试策略为编写Poem与toString方法的测试用例,对于Poem,采用不同的语料,测试能否得到正确的输出;对于toString,则用不同用例测试toString方法的正确性。

举个例子如图:在这里插入图片描述
mugar-omni-theater.txt中保存着语料

This is a test of the Mugar Omni Theater sound system.

此条测试的意义是输入句子input,经过图处理后,检验输出是否与我们的预期AccOutput相同。

3.具体实现Poet

在GraphPoet中,我们需要依次实现三个方法:

1) 构造方法GraphPoet

在GraphPoet的构造方法中,我们需要读取文本文档,并由其中语料按照前文所阐述的方法生成一个单词图。具体实现方式为:输入文件路径并按行读入,将单词进行切分,进行删除标点符号,大写转小写等预处理,每次取相邻元素在图中添加新边。

2) 生成句子方法Poem

Poem方法输入一个String参数作为原始句子,输出根据单词图匹配过的,扩充了bridge words的新句子。具体实现思路为一次读入两个单词,检索前一个单词子节点的子节点,若其中包含后一个单词,则选择途径通路权重最高的一路,将该路上途径节点表示的单词加入原来两单词之间。

3) toString方法

简单重写toString方法,将整个图中所有点的指向转化为一条字符串输出。

4.测试代码覆盖度

代码覆盖度建议尽量达到100%
在这里插入图片描述

四、重构Lab 1中的Social Network

这一环节要求我们基于在前面步骤中定义的Graph及其两种实现,将泛型L替换为Person,按照Lab1中Social NetWorek的要求,实现Lab1 Problem 3中FriendshipGraph的各种功能,并且尽可能复用我们在前文构造的类中已经实现的方法。最后,运行提供的main ( ) 和执行Lab1中的Junit测试用例,确保其正常运行。

有Lab 1的基础,这部分的重构算是比较快的。利用ConcreteEdgesGraphTestConcreteVerticesGraphTest其一便可完成。测试代码与main函数之间copy,略加修改就可。


总结

总得来说,Lab 2相较于前几届的版本,难度降低了很多,但仍有不易理解的地方,需要花一些时间才能完成。

这篇关于哈工大软件构造Lab2导读 - Stanford 6.031 Problem Set 2: Poetic Walks的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/561796

相关文章

Ubuntu 怎么启用 Universe 和 Multiverse 软件源?

《Ubuntu怎么启用Universe和Multiverse软件源?》在Ubuntu中,软件源是用于获取和安装软件的服务器,通过设置和管理软件源,您可以确保系统能够从可靠的来源获取最新的软件... Ubuntu 是一款广受认可且声誉良好的开源操作系统,允许用户通过其庞大的软件包来定制和增强计算体验。这些软件

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

uva 10025 The ? 1 ? 2 ? ... ? n = k problem(数学)

题意是    ?  1  ?  2  ?  ...  ?  n = k 式子中给k,? 处可以填 + 也可以填 - ,问最小满足条件的n。 e.g k = 12  - 1 + 2 + 3 + 4 + 5 + 6 - 7 = 12 with n = 7。 先给证明,令 S(n) = 1 + 2 + 3 + 4 + 5 + .... + n 暴搜n,搜出当 S(n) >=

软件设计师备考——计算机系统

学习内容源自「软件设计师」 上午题 #1 计算机系统_哔哩哔哩_bilibili 目录 1.1.1 计算机系统硬件基本组成 1.1.2 中央处理单元 1.CPU 的功能 1)运算器 2)控制器 RISC && CISC 流水线控制 存储器  Cache 中断 输入输出IO控制方式 程序查询方式 中断驱动方式 直接存储器方式(DMA)  ​编辑 总线 ​编辑

poj 3050 dfs + set的妙用

题意: 给一个5x5的矩阵,求由多少个由连续6个元素组成的不一样的字符的个数。 解析: dfs + set去重搞定。 代码: #include <iostream>#include <cstdio>#include <set>#include <cstdlib>#include <algorithm>#include <cstring>#include <cm

【STM32】SPI通信-软件与硬件读写SPI

SPI通信-软件与硬件读写SPI 软件SPI一、SPI通信协议1、SPI通信2、硬件电路3、移位示意图4、SPI时序基本单元(1)开始通信和结束通信(2)模式0---用的最多(3)模式1(4)模式2(5)模式3 5、SPI时序(1)写使能(2)指定地址写(3)指定地址读 二、W25Q64模块介绍1、W25Q64简介2、硬件电路3、W25Q64框图4、Flash操作注意事项软件SPI读写W2

免费也能高质量!2024年免费录屏软件深度对比评测

我公司因为客户覆盖面广的原因经常会开远程会议,有时候说的内容比较广需要引用多份的数据,我记录起来有一定难度,所以一般都用录屏工具来记录会议内容。这次我们来一起探索有什么免费录屏工具可以提高我们的工作效率吧。 1.福晰录屏大师 链接直达:https://www.foxitsoftware.cn/REC/  录屏软件录屏功能就是本职,这款录屏工具在录屏模式上提供了多种选项,可以选择屏幕录制、窗口

leetcode105 从前序与中序遍历序列构造二叉树

根据一棵树的前序遍历与中序遍历构造二叉树。 注意: 你可以假设树中没有重复的元素。 例如,给出 前序遍历 preorder = [3,9,20,15,7]中序遍历 inorder = [9,3,15,20,7] 返回如下的二叉树: 3/ \9 20/ \15 7   class Solution {public TreeNode buildTree(int[] pr

Collection List Set Map的区别和联系

Collection List Set Map的区别和联系 这些都代表了Java中的集合,这里主要从其元素是否有序,是否可重复来进行区别记忆,以便恰当地使用,当然还存在同步方面的差异,见上一篇相关文章。 有序否 允许元素重复否 Collection 否 是 List 是 是 Set AbstractSet 否

HomeBank:开源免费的个人财务管理软件

在个人财务管理领域,找到一个既免费又开源的解决方案并非易事。HomeBank&nbsp;正是这样一个项目,它不仅提供了强大的功能,还拥有一个活跃的社区,不断推动其发展和完善。 开源免费:HomeBank 是一个完全开源的项目,用户可以自由地使用、修改和分发。用户友好的界面:提供直观的图形用户界面,使得非技术用户也能轻松上手。数据导入支持:支持从 Quicken、Microsoft Money