天天看点

使用强大的 Mockito 测试框架来测试你的代码

<b>本文讲的是使用强大的 Mockito 测试框架来测试你的代码,</b>

<b></b>

这篇教程介绍了如何使用 Mockito 框架来给软件写测试用例

如果需要往下学习,你需要先理解 Junit 框架中的单元测试。

单元测试的思路是在不涉及依赖关系的情况下测试代码(隔离性),所以测试代码与其他类或者系统的关系应该尽量被消除。一个可行的消除方法是替换掉依赖类(测试替换),也就是说我们可以使用替身来替换掉真正的依赖对象。

dummy object 做为参数传递给方法但是绝对不会被使用。譬如说,这种测试类内部的方法不会被调用,或者是用来填充某个方法的参数。

Fake 是真正接口或抽象类的实现体,但给对象内部实现很简单。譬如说,它存在内存中而不是真正的数据库中。(译者注:Fake 实现了真正的逻辑,但它的存在只是为了测试,而不适合于用在产品中。)

stub 类是依赖类的部分方法实现,而这些方法在你测试类和接口的时候会被用到,也就是说 stub 类在测试中会被实例化。stub类会回应任何外部测试的调用。stub 类有时候还会记录调用的一些信息。

mock object 是指类或者接口的模拟实现,你可以自定义这个对象中某个方法的输出结果。

测试替代技术能够在测试中模拟测试类以外对象。因此你可以验证测试类是否响应正常。譬如说,你可以验证在 Mock 对象的某一个方法是否被调用。这可以确保隔离了外部依赖的干扰只测试测试类。

我们选择 Mock 对象的原因是因为 Mock 对象只需要少量代码的配置。

你可以手动创建一个 Mock 对象或者使用 Mock 框架来模拟这些类,Mock 框架允许你在运行时创建 Mock 对象并且定义它的行为。

一个典型的例子是把 Mock 对象模拟成数据的提供者。在正式的生产环境中它会被实现用来连接数据源。但是我们在测试的时候 Mock 对象将会模拟成数据提供者来确保我们的测试环境始终是相同的。

Mock 对象可以被提供来进行测试。因此,我们测试的类应该避免任何外部数据的强依赖。

通过 Mock 对象或者 Mock 框架,我们可以测试代码中期望的行为。譬如说,验证只有某个存在 Mock 对象的方法是否被调用了。

Mockito 是一个流行 mock 框架,可以和JUnit结合起来使用。Mockito 允许你创建和配置 mock 对象。使用Mockito可以明显的简化对外部依赖的测试类的开发。

一般使用 Mockito 需要执行下面三步

模拟并替换测试代码中外部依赖。

执行测试代码

验证测试代码是否被正确的执行

使用强大的 Mockito 测试框架来测试你的代码

如果你的项目使用 Gradle 构建,将下面代码加入 Gradle 的构建文件中为自己项目添加 Mockito 依赖

Eclipse IDE 支持 Gradle 和 Maven 两种构建工具,所以在 Eclipse IDE 添加依赖取决你使用的是哪一个构建工具。

在 Eclipse RCP 应用依赖通常可以在 p2 update 上得到。Orbit 是一个很好的第三方仓库,我们可以在里面寻找能在 Eclipse 上使用的应用和插件。

使用强大的 Mockito 测试框架来测试你的代码

如果在代码中静态引用了<code>org.mockito.Mockito.*;</code>,那你你就可以直接调用静态方法和静态变量而不用创建对象,譬如直接调用 mock() 方法。

除了上面所说的使用 mock() 静态方法外,Mockito 还支持通过 <code>@Mock</code> 注解的方式来创建 mock 对象。

如果你使用注解,那么必须要实例化 mock 对象。Mockito 在遇到使用注解的字段的时候,会调用<code>MockitoAnnotations.initMocks(this)</code> 来初始化该 mock 对象。另外也可以通过使用<code>@RunWith(MockitoJUnitRunner.class)</code>来达到相同的效果。

通过下面的例子我们可以了解到使用<code>@Mock</code> 的方法和<code>MockitoRule</code>规则。

告诉 Mockito 模拟 databaseMock 实例

Mockito 通过 @mock 注解创建 mock 对象

使用已经创建的mock初始化这个类

在测试环境下,执行测试类中的代码

使用断言确保调用的方法返回值为 true

验证 query 方法是否被 <code>MyDatabase</code> 的 mock 对象调用

当我们需要配置某个方法的返回值的时候,Mockito 提供了链式的 API 供我们方便的调用

<code>when(…​.).thenReturn(…​.)</code>可以被用来定义当条件满足时函数的返回值,如果你需要定义多个返回值,可以多次定义。当你多次调用函数的时候,Mockito 会根据你定义的先后顺序来返回返回值。Mocks 还可以根据传入参数的不同来定义不同的返回值。譬如说你的函数可以将<code>anyString</code> 或者 <code>anyInt</code>作为输入参数,然后定义其特定的放回值。

对于无返回值的函数,我们可以使用<code>doReturn(…​).when(…​).methodCall</code>来获得类似的效果。例如我们想在调用某些无返回值函数的时候抛出异常,那么可以使用<code>doThrow</code> 方法。如下面代码片段所示

Mockito 会跟踪 mock 对象里面所有的方法和变量。所以我们可以用来验证函数在传入特定参数的时候是否被调用。这种方式的测试称行为测试,行为测试并不会检查函数的返回值,而是检查在传入正确参数时候函数是否被调用。

@Spy或者<code>spy()</code>方法可以被用来封装 java 对象。被封装后,除非特殊声明(打桩 stub),否则都会真正的调用对象里面的每一个方法

方法<code>verifyNoMoreInteractions()</code>允许你检查没有其他的方法被调用了。

我们也可以使用<code>@InjectMocks</code> 注解来创建对象,它会根据类型来注入对象里面的成员方法和变量。假定我们有 ArticleManager 类

这个类会被 Mockito 构造,而类的成员方法和变量都会被 mock 对象所代替,正如下面的代码片段所示:

创建ArticleManager实例并注入Mock对象

<code>ArgumentCaptor</code>类允许我们在verification期间访问方法的参数。得到方法的参数后我们可以使用它进行测试。

Mockito当然也有一定的限制。而下面三种数据类型则不能够被测试

final classes

anonymous classes

primitive types

在 Android 中的 Gradle 构建文件中加入 Mockito 依赖后就可以直接使用 Mockito 了。若想使用 Android Instrumented tests 的话,还需要添加 dexmaker 和 dexmaker-mockito 依赖到 Gradle 的构建文件中。(需要 Mockito 1.9.5版本以上)

创建一个包名为<code>com.vogella.android.testing.mockito.contextmock</code>的Android应用,添加一个静态方法 ,方法里面创建一个包含参数的Intent,如下代码所示:

使用 Mockito 创建一个单元测试来验证在传递正确 extra data 的情况下,intent 是否被触发。

因此我们需要使用 Mockito 来 mock 一个<code>Context</code>对象,如下代码所示:

创建一个 Api,它可以被 Mockito 来模拟并做一些工作

实现 <code>TwitterClient</code>类,它内部使用到了 <code>ITweet</code> 的实现。但是<code>ITweet</code>实例很难得到,譬如说他需要启动一个很复杂的服务来得到。

为了能够不启动复杂的服务来得到 <code>ITweet</code>,我们可以使用 Mockito 来模拟得到该实例。

现在 <code>TwitterClient</code> 可以使用 <code>ITweet</code> 接口的实现,当调用 <code>getMessage()</code> 方法的时候将会打印 "Using Mockito is great" 信息。

确保 getMessage() 方法至少调用一次。

运行测试,查看代码是否测试通过。

因为 Mockito 不能够 mock 静态方法,因此我们可以使用 <code>Powermock</code>。

我们模拟了 NetworkReader 的依赖,如下代码所示:

有时候我们可以在静态方法周围包含非静态的方法来达到和 Powermock 同样的效果。

<b>原文发布时间为:2016年07月20日</b>

<b>本文来自云栖社区合作伙伴掘金,了解相关信息可以关注掘金网站。</b>

继续阅读