`
jiangduxi
  • 浏览: 446390 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
社区版块
存档分类
最新评论

怎么抽取一个测试模块?

阅读更多
[/b][b]问题:
   当为一个产品类编写了好几个测试,它们含有重复的代码。因为我们知道重复的代码是软件中很多问题的根源,怎么 消除代码中的重复性。

背景:
    为同一个产品类编写几个测试的时候,你最先注意到的模式之一就是每个测试开始的那几行代码总是很相像。每个测试都有三个基本的组成部分:创建一个对象,调用一些方法,检查结果。每个测试的第二部分总是不同的,对不同方法的调用可以区别不同的测试:"如果用这些参数调用构造方法,希望看到这样的结果;但是如果传递了空值,那么构造方法应该抛出那样的异常"。
   测试的第3部分,检查结果,这完全依赖于你所调用的方法,如果测试调用的是不同的方法,当然期望的结果也是不同的。一般情况下,在这部分,只有在反复调用同一个方法的时候才会有重复代码出现。

  然而,创建对象会产生测试之间的重复代码,而且绝大部分重复的代码都出现在这里,在一个类中除了构造方法以外还有很多方法需要测试,所以一旦为这个类编写了第二个测试,就重复了第一个测试中"创建一些对象"的部分,因为很可能用相同的参数调用了相同的构造方法,这样的重复代码实在是很常见,所以如果Junit能有一种内置的机制来消除它,那就再好不过了。

那么如果将要测试的对象称为测试模块:一种对象的"配置",其行为是可以预见的。这样的话,我们可以将测试第一部分----"创建一些对象"称为“创建一个测试模块”。目的是创建一些对象并且将其初始化成某种已知的状态,以便在使用它们调用一些方法的时候,可以预见到结果。

技巧:
   从测试中找出重复的测试模块代码,将这些代码移动到一个叫做setUp()的方法里面。修改后的代码可能不需要编译,因为现在是在setUp()方法中声明变量,然后在测试中使用这些变量。将这些变量放在实例级的参数域中,这样setUp()和测试代码都可以使用它们。因为每个测试是执行在其自己的实例中,所以不必担心在不同测试之间实例级的参数域会被错误地改变。在执行测试的时候,test runner会在每个测试之前调用setUp()方法。测试类中的每个测试都可以使用这个通用的方法初始化。

代码演示:
下面代码Money JavaBean及方法就不给出,可以根据测试代码,自己推出
   public class MoneyTest extends TestCase{
   
      public void testAdd(){
           Money added = new Money(12,50);
           Money augend = new Money(12,50);
           Money sum = added.add(augend);
           assertEquals(2500,sum.inCents());
         }
      public void testNegate(){
          Money money = new Money(12,50);
          Money opposite = money.negate();
          assertEquals(-1250,opposite.inCents());
        }
     
      public void testRound(){
         Money money = new Money(12,50);
         Money rounded = money.roundToNearestDollar();
         assertEquals(1300,rounder.inCents());
        }
    }

注意上面的三个测试的第一行几乎是一样的,似乎可以将这个对象移到setUp()方法里。因为testAdd()方法将这个对象叫做added而其他叫money,首先需要将added相关money以使得三个测试的第一行都一样,然后就可以将这一行移到setUp()方法里去。最后,将money从一个本地变量变成一个实例级的参数域,使得setUp()和所有的测试都可以使用它。接着看看改动后的代码:
   public class MoneyTest extend TestCase{
       private Money money;
 
       protected void setUp()throws Exception{
          money = new Money(12,50);
       }

       public void testAdd(){
         Money sum = added.add(augend);
         assertEquals(2500,sum.inCents());

       }
       
       public void testNegate(){
          Money opposite = money.negate();
          assertEquals(-1250,opposite.inCents());
        }
     
      public void testRound(){
         Money rounded = money.roundToNearestDollar();
         assertEquals(1300,rounder.inCents());
        }

     }

上面这样的做,我们其实在数学中有个概念很好的用来说下:那就是将方程式进行变形然后提取公因式。任何东西都是可以进行检讨和讨论

讨论:
   Junit通过方法setUp()和teatDown()提供了对测试模块的支持,可以在junit.framework.TestCase中找到他们。在创建TestCase的子类的时候,可以重载这些方法来为每个测试建立或者销毁测试模块。要了解Junit是如何使用这些方法的,可以查看另一个TestCase方法runBare()的代码:
  public void runBare()throws Throwable{
       setUp();
   try{
      runTest();
        }
   finally{
         tearDown();
       }
   }

在执行测试的时候,框架会首先调用runBare()方法来设置一个测试模块,运行测试,然后再撤销测试模块。注意将tearDown()放在finally块中,就能保证它一定被执行,即使测试失效,这个方法是为了释放资源,避免资源浪费。
分享到:
评论

相关推荐

    Python 随机生成测试数据的模块:faker基本使用方法详解

    faker是python的一个第三方模块,是一个github上的开源项目。 主要用来创建一些测试用的随机数据。 官方文档:https://faker.readthedocs.io/en/master/index.html faker的使用: 1.安装模块 pip3 install Faker...

    FPGA实现FFT的论文

    整个解调模块的测试以及验证过程。解调模块的设计采用的是基 -2 按时间抽取的 方法来实现的。以 altera 公司提供的 quartus ii 11.0 为软件平台,利用 VHDL 描 述的方式实现了 12 位 512 点的 FFT 解调,并且...

    C#项目开发案例全程实录(第2版)下载地址

     本书光盘提供12个项目开发完整案例的同步教学视频,12个项目开发完整案例及其源程序,本书的服务网站还提供了内容丰富的素材库、题库、模块库、案例库和本书的答疑服务等。  本书是一本项目开发案例方面的参考书...

    MMLAB实战系列视频教程|MMLAB视频教程

    第3章 第一模块:训练结果测试与验证 第4章 第一模块:模型源码DEBUG演示 第5章 第二模块:使用分割模块训练自己的数据集 第6章 第二模块:基于Unet进行各种策略修改 第7章 第二模块:分割任务CVPR最新Backbone设计...

    论文研究-基于SPARDL的模型和程序一致性测试.pdf

    提出基于SPARDL(space aircraft description language)的模型和程序一致性测试方法, 通过模型抽取获取模式迁移图和控制流程图, 通过程序插桩获取程序执行路径, 实现了自动检测周期控制系统中的模式迁移和模块调用的...

    MMLAB实战系列视频教程

    第3章 第一模块:训练结果测试与验证 第4章 第一模块:模型源码DEBUG演示 第5章 第二模块:使用分割模块训练自己的数据集 第6章 第二模块:基于Unet进行各种策略修改 第7章 第二模块:分割任务CVPR最新Backbone设计...

    Java毕业设计:基于SpringBoot的有声书城应用设计与实现(源码+文档+PPT+录像演示).zip

    有声书城 SpringBoot;JQuery;MYSQL 基于JSP的有声书城专门为那些喜欢在线阅读的人士服务,除了可以在线进行阅读之外,书城还有提供...确定了模块以后,再设计各个模块的功能特点,方便抽取出各个模块的公共部分来。

    基于SpringBoot的有声书城应用设计与实现.zip

    有声书城 SpringBoot;JQuery;MYSQL 基于JSP的有声书城专门为那些喜欢在线阅读的人士服务,除了可以在线进行阅读之外,书城还有提供...确定了模块以后,再设计各个模块的功能特点,方便抽取出各个模块的公共部分来。

    在线考试网源码,测试可用

    在抽取试题上,系统使用随机抽取试题的方式,体现了考试的客观与公正。 当考生答题完毕之后,提交试卷即可得知本次考试的得分,体现系统的高效性。在后台管理上,分后台管理员管理模块和试题管理模块。 分别适应...

    论文研究-统计机器翻译中翻译规则抽取.pdf

    对齐短语是决定统计机器翻译系统质量的核心模块。提出基于短语结构树的层次短语模型,这是利用串-树模型的思想对层次短语模型的扩展。基于短语结构树的层次短语模型是在双语对齐短语的基础之上结合英语短语结构树...

    论文研究-混合的本体原子分解方法.pdf

    提出一种混合的原子分解算法,首先利用有向超图表示EL子本体,形成部分原子分解,利用模块抽取方法添加剩余非EL公理,得到本体的全部原子分解。以生物医学本体作为测试数据,实验表明,这种混合的原子分解算法能够...

    大数据应用测试经验总结.pdf

    ⼤数据应⽤测试经验总结 ⼤数据应⽤测试经验总结 ⼤数据应⽤测试过程与传统的web系统有较⼤的不同,⼤数据应⽤测试通常会分为web侧和ETL侧测试,web侧基本就是功能测试,⽽ETL(Extracting-Transfroming- Loading)...

    weixin052用于日语词汇学习的微信小程序+ssm(源码+部署说明+演示视频+源码介绍+lw).rar

    SpringMVC:SpringMVC是Spring的一个模块,它是一个基于Java的实现了MVC设计模式的请求驱动类型的轻量级Web框架。 MyBatis:MyBatis是一个优秀的持久层框架,它支持定制化SQL、存储过程以及高级映射。 微信小程序...

    软件工程-理论与实践(许家珆)习题答案

    ● 测试(发现并改正错误,分为模块测试、集成测试和系统联调三级); ● 运行维护(扩充功能、纠错等)。 习题二答案 一、 选择题 1. 需求分析的主要目的是(B C)。 A) 系统开发的具体方案 B) 进一步确定用户的...

    服务虚拟化Terminator.zip

    在开发/测试一个复杂系统的时候我们经常遇到开发/测试中的模块依赖其它服务的情况。比如一个系统有两个模块A和B,A模块依赖于B模块提供的服务: B部分功能还未完成开发导致A模块开发被阻塞; B模块有些数据不好...

    计算机二级公共基础知识

    在树结构中,每一个结点只有一个前件,称为父结点,没有前件的结点只有一个,称为树的根结点,简称树的根。例如,在图1-1中,结点A是树的根结点。 子结点和 叶子结点 在树结构中,每一个结点可以有多个后件,称为该...

    综合电子设备管理系统的可维护性设计.doc

    一个源代码包含多个相关功能代码。在二期系统中 ,我们改善了子系统的设计,增强各个模块的内聚性,降低模块之间的耦合性。首先, 各个功能的相关代码处于不同的代码目录;其次,各模块都做成了单独的静态库,总控 ...

    综合电子设备管理系统的可维护性设计(1).doc

    一个源代码包含多个相关功能代码。在二期系统中 ,我们改善了子系统的设计,增强各个模块的内聚性,降低模块之间的耦合性。首先, 各个功能的相关代码处于不同的代码目录;其次,各模块都做成了单独的静态库,总控 ...

    北邮 基于三层交换机完成校园网搭建 实验报告.docx

    主要测试模块包括以下几点: 1. 通过VLAN的划分抑制广播风暴,并实现不同VLAN间相互通信的需求。 2. 三层交换机各端口设置参数及其功能特点。 3. 通过静态路由、RIP协议实现各VLAN间的相互通信。 在此基础上总结并...

    query_analysis:自然语言处理(NLP) 语句抽取

    当前业界文本语义信息抽取基本是使用正则表达式来完成的,正则表达式本身并不提供管理与测试功能。 在进行基于正则的文本语义抽取的时候,还需要做大量的基础工作,来便于编写、管理和测试正则表达式。 本项目提供...

Global site tag (gtag.js) - Google Analytics