文章目录
- 引言
- Mock 的定义
- Mock 的分类
- Mock 一个对象
- Mock Server
- 小结
引言
实际工作中,测试人员可能会遇到如下情况:
- 场景一:依赖接口不通,甲开发A模块,乙开发B模块,甲的进度比乙快,但A模块的方法依赖于B模块,要测试A模块接口怎么办?
- 场景二:异常数据难模拟,当需要测试接口一些异常数据,接口正常情况是否无法提供异常数据的。那么如何简便地构造接口的异常数据?
- 场景三:依赖接口性能参数无法保障。在对接口性能压测的时候,需要下游接口及时返回数据,满足上游接口的调用频度。在依赖接口多的情况下,如何减轻工作量?
Mock 的定义
在性能基础之浅谈常见接口性能压测文中,我们有简单介绍过 Mock ,本文将详细阐述 Mock 概念。
在接口测试过程中,对于某些不容易构造或者不容易获取的对象,我们常常会用一个虚拟的对象代替以便测试。在具体的测试过程中,我们经常会碰到需要模拟数据或者接口的情况,因为环境问题或者系统复杂度的问题,我们需要使用 Mock 方式进行数据的模拟。
引用淘宝网《接口测试白皮书》中的对 Mock 的定义:
Mock 是指使用各种技术手段模拟出各种需要的资源以供测试使用。
被 Mock 的资源通常有以下特征:
- 被测目标依赖该资源
- 该资源可能因为各种原因不稳定、返回结果不断变化或者并不总是能够获取到
- 该资源跟被测目标本身质量无关
- 这些资源可能是一个外部或底层接口、一个系统、一组数据对象或者是一整套目标软件的工作环境等。通过 Mock 避免对外部真实资源的依赖实现对被测目标的孤立测试,从而大大降低测试的难度,节约测试成本。
- 需要注意的是利用 Mock 通过的测试与使用真实环境通过的测试毕竟还是有一定差别的。有些时候我们就是需要所测试的系统能够处理依赖所产生的各种情况,包括正常情况和异常情况,我们同样不能保证我们的Mock 可以模拟到每种这样的情况。因此只在确实有必要的情况下才运用Mock。
Mock 的分类
目前主要应用两大类 Mock 的场景。
一种是 Mock 一个对象,写入一些预期的值,通过它进行自己想要的测试。主要适用于单元测试,哪种语言开发的程序必须用基于哪种语言的Mock 方案去实现。
例如:Mockito 只能针对 Java ,适用范围:单测
另外一种就是 Mock 一个 Server ,构造一个假的服务返回预期的结果,也是为了进行自己的测试。主要适用于接口&性能测试,Mock 方案和程序使用的语言无关,可以用 Java 实现,也可以用 Python 实现等,例如:搭建一个 Mock Server,适用范围:无限制
这两个场景构造了大部分的 Mock 使用范围。
Mock 一个对象
此处使用Mockito示例
Mockito 是 GitHub 上使用最广泛的 Mock 框架,并与 JUnit 结合使用。Mockito 框架可以创建和配置 mock 对象。使用 Mockito 简化了具有外部依赖的类的测试开发。
一般使用 Mockito 的步骤:
- 模拟任何外部依赖并将这些模拟对象插入测试代码中
- 执行测试中的代码执行测试中的代码
- 验证代码是否按照预期执行验证代码是否按照预期执行
引入pom
<dependency>
<groupId>org.mockito</groupId>
<artifactId>mockito-all</artifactId>
<version>1.10.19</version>
</dependency>
<dependency>
<groupId>org.testng</groupId>
<artifactId>testng</artifactId>
<version>6.14.3</version>
</dependency>
public class SimpleTest {
@Test
public void test(){
// 创建Mock对象,参数可以是类或者接口
List<String> list = mock(List.class);
//设置方法的预期返回值
when(list.get(0)).thenReturn("zuozewei");
when(list.get(1)).thenThrow(new RuntimeException("test exception"));
String result = list.get(0);
//验证方法调用
verify(list).get(0);
//断言,list的第一个元素是否是"zuozwei"
Assert.assertEquals(result,"zuozewei");
}
}
新建测试类,构造了 list 这样的对象,并且给一个元素赋值 zuozewei。在最后断言的时候,也可以通过这个 list 里面确实有这个值。所以,通过这种方式,我们可以进行对象构造。可以是类,也可以是接口。
除了构造对象,当然也可以对方法设定的返回值指定异常。
上述代码的意思就是当调用 list 的第二个元素的时候,抛出一个运行异常。
public class SimpleTest {
<span class="token annotation punctuation">@Test</span>
<span >public</span> <span >void</span> <span class="token function">test</span><span >(</span><span >)</span><span >{</span>
<span class="token comment">// 创建Mock对象,参数可以是类或者接口</span>
List<span class="token generics function"><span ><</span>String<span >></span></span> list <span class="token operator">=</span> <span class="token function">mock</span><span >(</span>List<span >.</span><span >class</span><span >)</span><span >;</span>
<span class="token comment">//设置方法的预期返回值</span>
<span class="token function">when</span><span >(</span>list<span >.</span><span class="token function">get</span><span >(</span><span class="token number">0</span><span >)</span><span >)</span><span >.</span><span class="token function">thenReturn</span><span >(</span><span class="token string">"zuozewei"</span><span >)</span><span >;</span>
<span class="token function">when</span><span >(</span>list<span >.</span><span class="token function">get</span><span >(</span><span class="token number">1</span><span >)</span><span >)</span><span >.</span><span class="token function">thenThrow</span><span >(</span><span >new</span> <span >RuntimeException</span><span >(</span><span class="token string">"test exception"</span><span >)</span><span >)</span><span >;</span>
String result <span class="token operator">=</span> list<span >.</span><span class="token function">get</span><span >(</span><span class="token number">0</span><span >)</span><span >;</span>
<span class="token comment">//验证方法调用</span>
<span class="token function">verify</span><span >(</span>list<span >)</span><span >.</span><span class="token function">get</span><span >(</span><span class="token number">0</span><span >)</span><span >;</span>
<span class="token comment">//断言,list的第一个元素是否是"zuozwei"</span>
Assert<span >.</span><span class="token function">assertEquals</span><span >(</span>result<span >,</span><span class="token string">"zuozewei"</span><span >)</span><span >;</span>
<span >}</span>
}
上面只是列举了 Mockito 的简单用法。对于比较复杂的用法,大家可以通过官网深入学习。因为 Mockito 主要用于单元测试,开发人员用的比较多,所以大家有兴趣可以自行了解。
Mock Server
下图很好的解释了Mock Server 位置和作用: