软件测试技术实战 设计、工具及管理》联载-32

简介: 软件测试技术实战 设计、工具及管理》联载-32

8.1.3  对代码的详细介绍


一个单元测试代码主要分为以下几个部分。

1)设置环境。

2)运行测试。

3)结果判断。

4)清理环境。

这里,设置环境和清理环境是有区别的。

设置环境:比如,建立数据库连接。

清理环境:比如,断开数据库连接。


1.包含必要的Package

importstatic org.junit.Assert.*;

assertEqualsAssert类中的一系列的静态方法,一般的使用方式是Assert. assertEquals(),但是使用了静态包后,前面的类名就可以省略了,使用起来更加方便。比如:

assertEquals(8, calculator.getResult());

显而易见,assertEquals函数的主要功能是实现“结果判断”。


2.测试类的声明

测试类是一个独立的类,没有任何父类。测试类的名字也可以任意命名,没有任何局限性。所以,不能通过类的声明来判断它是不是一个测试类。测试类与普通类的区别在于它内部方法的声明。


3.创建一个待测试的对象

要测试哪个类,要创建一个该类的对象。比如第8.1.1节和第8.1.2节中,为了测试Calculator类,必须创建一个Calculator对象。

privatestatic Calculator calculator = new Calculator();


4.测试方法的声明

在测试类中,并不是每个方法都用于测试,必须使用“标注”来明确表明哪些是测试方法。“标注”也是JDK5的一个新特性,用在此处非常恰当。可以看到,某些方法的前面有@Before@Test@Ignore@After等字样,这些就是标注,以一个“@”作为开头。第8.1.3节开始的描述的@Before@Test@After对应于:

1)标记@Before:设置环境。

2)标记@Test:运行测试。

3)标记@After:清理环境。

这个方法的前面使用@Test标注,表明这是一个测试方法。对方法的声明,也有如下要求:名字可以随便取,没有任何限制,但是返回的值必须为void,而且不能有任何参数。如果违反这些规定,会在运行时抛出一个异常。


@Test
publicvoid testSubstract() {
        calculator.substract(10,2);
        assertEquals(8,calculator.getResult());
}


在测试方法中调用substract函数,将10减去2,期待的结果应该是8。如果最终的实际结果也是8,则说明substract函数是正确的,反之说明它是错的。


assertEquals(8,calculator.getResult()):用来判断期待结果和实际结果是否相等,第一个参数填写期待结果,第二个参数填写实际结果,也就是通过计算得到的结果。这样写好后,JUnit会自动进行测试,并把测试结果反馈给用户。


8.1.4  对JUnit 4的高级操作


1@BeforeClass @AfterClass

有一个类是负责对大文件(超过500MB)进行读写,它的每一个方法都是对文件进行操作。换句话说,调用每个方法前,都要打开一个大文件,并读入文件内容,这绝对是一个非常耗时的操作。如果使用@Before@After,那么每次测试都要读取一次文件,效率极其低下。


@BeforeClass@AfterClass两个标识来帮助实现这个功能。从名字上就可以看出,用这两个Fixture标注的函数,只在测试用例初始化时执行 @BeforeClass方法,当所有测试执行完毕后,执行@AfterClass方法进行收尾工作。这里要注意,每个测试类只能有一个方法被标注为@BeforeClass@AfterClass,并且该方法必须是publicstatic


2.防止超时

比如,程序里存在死循环,如何处理?如在“简单计算器”产品代码中,有语句:

publicvoid squareRoot( int n)  {
   for(; ;) ;                  // Bug : 死循环
}


要实现这一功能,给@Test标注加一个参数即可:

@Test(timeout=1000) //1000ms
publicvoid testSquareRoot()  {
    calculator.squareRoot( 4 );
    assertEquals( 2,calculator.getResult());
}


也就是说,测试用例等待时间1000ms(即1s),如果1s内没有反应,就认为测试失败。


3Runner(运行器)

当测试代码提交给JUnit 4框架后,JUnit 4框架通过Runner如何来运行测试代码。如果不设置,就认为是默认的,但是也可以设置:


import  org.junit.internal.runners.TestClassRunner;
import  org.junit.runner.RunWith;
// 使用了系统默认的TestClassRunner,与下面代码完全一样
public classCalculatorTest  {
  ...
  }
@RunWith(BlockJUnit4ClassRunner.class)
 public class CalculatorTest  {
 ...
}


JUnit 4Runner主要有BlockJUnit4ClassRunnerParentRunnerStatementTestRuleDescriptionRunNotifierEnclosedInvokeMethod。其中BlockJUnit4ClassRunner.class是默认的Runner

l Enclosed:是实现内部类中测试类的运行器。

l ParentRunner:是JUnit 4测试执行器的基类,它提供了一个测试器所需要的大部分功能。继承它的类需要实现:

protectedabstract List<T> getChildren();

protectedabstract Description describeChild(T child);

protectedabstract void runChild(T child,RunNotifier notifier);

l Parameterized:则可以设置参数化测试用例。

l JUnit38ClassRunner:是为了向后兼容JUnit 3而定义的运行器。

l Statement:在运行时,执行test case前可以插入一些用户动作,它就是描述这些动作的一个类。继承这个类要实现:

 

/**
        *Run the action,throwing a {@code Throwable} ifanything goes wrong.
        */
    publicabstract void evaluate() throws Throwable;


这个方法会先后在ParentRunner.run()ParentRunner.runLeaf()这两个方法里面调用。另外,我们可以自定义一个Statement,并且实现evaluate()方法。


l TestRuleTestRule可以描述一个或多个测试方法如何运行和报告信息的接口。在TestRule中可以额外加入一些check,我们可以让一个test case失败/成功,也可以加入一些setupcleanup要做的事,也可以加入一些log之类的报告信息。总之,跑test case之前的任何事,都可以在里面做。需要实现apply()方法。

 

/**
    * Modifies the method-running{@link Statement} to implement this
    * test-running rule.
    * @param base The {@linkStatement} to be modified
    * @param description A {@linkDescription} of the test implemented in {@code base}
    * @return a new statement,which may be the same as {@codebase},
    * a wrapper around {@code base},or a completely new Statement.
    */
    Statement apply(Statementbase,Description description);


l Description:存储着当前单个或多个test case的描述信息。这些信息跟逻辑无关,比如原数据信息等。实例化DescriptionDescription.createTestDescription()方法。

l RunNotifier:运行时通知器。执行Runner.run(RunNotifierrunNotifier)方法时,需要传一个RunNotifier进去,这个RunNotifier是事件的管理器,它能帮助我们监控测试执行的情况。

l InvokeMethod:最终执行test case里面的测试方法通过这个类来做,这个类会间接调用Method.invoke()方法通知编译器执行@test方法。

l ……


4.参数化测试

案例8-2:计算一个数的平方。

测试“计算一个数的平方”这个函数,暂且分3类:正数、0、负数。测试代码如下:

import  org.junit.AfterClass;
import  org.junit.Before;
import  org.junit.BeforeClass;
import  org.junit.Test;
import  static org.junit.Assert. * ;
publicclass AdvancedTest  {
privatestatic Calculator calculator=newCalculator();
@Before
publicvoid clearCalculator()  {
    calculator.clear();     
}
@Test
       public void square1()  {
          calculator.square( 2 );
          assertEquals( 4,calculator.getResult());
      }
     @Test
       Public void square2()  {
          calculator.square( 0 );
          assertEquals( 0,calculator.getResult());
      }
    @Test
       Public void square3()  {
          calculator.square( - 3 );
          assertEquals( 9,calculator.getResult());
      }
  }


如果用参数化实现代码,就简化成如下形式:


import  static org.junit.Assert.assertEquals;
import  org.junit.Test;
import  org.junit.runner.RunWith;
import  org.junit.runners.Parameterized;
import  org.junit.runners.Parameterized.Parameters;
import  java.util.Arrays;
import  java.util.Collection;
@RunWith(Parameterized.class )
public  class SquareTest{
private  static Calculator calculator = new Calculator();
private  int param;
private  int result;
@Parameters
       public static Collection<Object[]>data {
           return  Arrays.asList( new  Object[][] {
                   {2,4},
                   {0,0},
                   {-3,9},
          } );
      }
// 构造函数,对变量进行初始化
       public void SquareTest(int param,int result) {
           this .param=param;
           this .result =result;
      }
      @Test
Public  void square()  {
          calculator.square(param);
          assertEquals(resul,calculator.getResult());
      }
  }


5.打包测试

对于打包测试,就是把所有的测试类打成一个包一起运行,其代码如下:

import  org.junit.runner.RunWith;
import  org.junit.runners.Suite;
@RunWith(Suite.class)
@Suite.SuiteClasses({
         CalculatorTest.class,
         SquareTest.class
         } )
public  class AllCalculatorTests{
}

 

顾翔凡言:

不是好的工作会给你带来好的心情,而是好的心情会给你带来好的工作。

目录
相关文章
|
1月前
|
数据采集 监控 机器人
浅谈网页端IM技术及相关测试方法实践(包括WebSocket性能测试)
最开始转转的客服系统体系如IM、工单以及机器人等都是使用第三方的产品。但第三方产品对于转转的业务,以及客服的效率等都产生了诸多限制,所以我们决定自研替换第三方系统。下面主要分享一下网页端IM技术及相关测试方法,我们先从了解IM系统和WebSocket开始。
60 4
|
2月前
|
测试技术 持续交付 UED
软件测试的艺术:确保质量的实战策略
在软件开发的舞台上,测试是那把确保每个功能如交响乐般和谐奏响的指挥棒。本文将深入探讨软件测试的重要性、基本类型以及如何设计高效的测试策略。我们将通过一个实际的代码示例,展示如何运用这些策略来提升软件质量和用户体验。
|
1月前
|
Java 测试技术 数据安全/隐私保护
软件测试中的自动化策略与工具应用
在软件开发的快速迭代中,自动化测试以其高效、稳定的特点成为了质量保证的重要手段。本文将深入探讨自动化测试的核心概念、常见工具的应用,以及如何设计有效的自动化测试策略,旨在为读者提供一套完整的自动化测试解决方案,帮助团队提升测试效率和软件质量。
|
2月前
|
前端开发 JavaScript 测试技术
前端测试技术中,如何提高集成测试的效率?
前端测试技术中,如何提高集成测试的效率?
|
1月前
|
Web App开发 IDE 测试技术
Selenium:强大的 Web 自动化测试工具
Selenium 是一款强大的 Web 自动化测试工具,包括 Selenium IDE、WebDriver 和 Grid 三大组件,支持多种编程语言和跨平台操作。它能有效提高测试效率,解决跨浏览器兼容性问题,进行性能测试和数据驱动测试,尽管存在学习曲线较陡、不稳定等缺点,但其优势明显,是自动化测试领域的首选工具。
224 17
Selenium:强大的 Web 自动化测试工具
|
2月前
|
机器学习/深度学习 人工智能 算法
BALROG:基准测试工具,用于评估 LLMs 和 VLMs 在复杂动态环境中的推理能力
BALROG 是一款用于评估大型语言模型(LLMs)和视觉语言模型(VLMs)在复杂动态环境中推理能力的基准测试工具。它通过一系列挑战性的游戏环境,如 NetHack,测试模型的规划、空间推理和探索能力。BALROG 提供了一个开放且细粒度的评估框架,推动了自主代理研究的进展。
59 3
BALROG:基准测试工具,用于评估 LLMs 和 VLMs 在复杂动态环境中的推理能力
|
2月前
|
监控 测试技术 开发工具
移动端性能测试工具
移动端性能测试工具
73 2
|
2月前
|
安全 前端开发 测试技术
如何选择合适的自动化安全测试工具
选择合适的自动化安全测试工具需考虑多个因素,包括项目需求、测试目标、系统类型和技术栈,工具的功能特性、市场评价、成本和许可,以及集成性、误报率、社区支持、易用性和安全性。综合评估这些因素,可确保所选工具满足项目需求和团队能力。
|
2月前
|
安全 网络协议 关系型数据库
最好用的17个渗透测试工具
渗透测试是安全人员为防止恶意黑客利用系统漏洞而进行的操作。本文介绍了17款业内常用的渗透测试工具,涵盖网络发现、无线评估、Web应用测试、SQL注入等多个领域,包括Nmap、Aircrack-ng、Burp Suite、OWASP ZAP等,既有免费开源工具,也有付费专业软件,适用于不同需求的安全专家。
366 2
|
2月前
|
监控 网络协议 Java
一些适合性能测试脚本编写和维护的工具
一些适合性能测试脚本编写和维护的工具