java mockito工具
Mockito 中文文档
本文档参考hehonghui/mockito-doc-zh项目,如有侵权,请联系删除
Mockito框架官方地址mockito,文档地址。
Mockito库能够Mock对象、验证结果以及打桩(stubbing)。
该文档您也可以通过http://mockito.org获取到。所有文档都保存在javadocs中,因为它能够保证文档与源代码的一致性。这样也能够让离线的用户从IDE直接访问到文档。这样一来也能够激励Mockito开发者在每次写代码、每次提交时更新对应的文档。
目录
- 迁移到Mockito 2.0
0.1 Mockito支持Android
0.2 免配置内联mock making - 验证某些行为
- 如何做一些测试桩 (Stub)
- 参数匹配器 (matchers)
- 验证函数的确切、最少、从未调用次数
- 通过打桩为无返回值函数抛出异常
- 验证执行执行顺序
- 确认交互(interaction)操作没有在mock对象上执行
- 查找多余的调用
- 简化mock对象的创建 - @Mock注解
- 为连续的调用打桩 (迭代器风格的打桩)
- 通过回调方式来打桩
- doReturn()、doThrow()、doAnswer()、doNothing()、doCallRealMethod()系列方法
- 监控真实对象
- 修改没有测试桩的调用的默认返回值 ( 1.7版本之后 )
- 为下一步的断言捕获参数 (1.8版本之后)
- 真实的局部模拟对象(mocks) (1.8版本之后)
- 重置mocks对象 (1.8版本之后)
- 故障排查与验证框架的使用 (1.8版本之后)
- 行为驱动开发的别名 (1.8版本之后)
- 可序列化的mocks(1.8.1版本之后)
- 新的注解 : @Captor,@Spy,@InjectMocks (1.8.3版本之后)
- 带超时的验证 (1.8.5版本之后)
- 自动实例化被@Spies, @InjectMocks注释的字段以及构造函数注入 (1.9.0版本之后)
- 单行测试桩 (1.9.0版本之后)
- 验证被忽略的测试桩 (1.9.0版本之后)
- mock详情 (2.2.x中改进)
- 真实实例的委托调用 (1.9.5版本之后)
- MockMaker API (1.9.5版本之后)
- BDD风格的验证 (1.10.0版本之后)
- 监视 或 模拟 抽象类 (1.10.12版本加入,在2.7.13 和 2.7.14版特征得到增强))
- Mockito的模拟对象 可以通过 classloaders 序列化/反序列化(1.10.0版本之后)
- Deep stubs 更好的泛型支持 (1.10.0版本之后)
- Mockito JUnit 规则 (1.10.17版本之后)
- 开启和关闭插件的开关 (1.10.15版本之后)
- 自定义验证失败消息 (2.1.0版本之后)
- Java8 Lambda匹配器的支持 (2.1.0版本之后)
- Java8 自定义Answer的支持 (2.1.0版本之后)
- 元信息和泛型信息保留 (2.1.0版本之后)
- Mocking final类型,枚举和final方法 (2.1.0版本之后)
- “严格的”Mocktio能提高生产效率并使测试用例更清晰(2.+版本之后)
- 框架集成的高级公开API (2.10.+版本之后)
- 集成新的API: 监听验证开始(verification start)事件(2.11.+版本之后)
- 集成新的API: 测试框架支持MockitoSession(2.15.+版本之后)
- org.mockito.plugins.InstantiatorProvider泄露内部API所以被org.mockito.plugins.InstantiatorProvider2替代(2.15.4版本之后)
- JUnit5+的扩展
- 新的Mockito.lenient()和MockSettings.lenient()方法(2.20.0版本之后)
0. 迁移到Mockito 2.0
为了持续提升Mockito以及更进一步的提升单元测试体验,我们希望你升级到Mockito 2.0.Mockito遵循语意化的版本控制,除非有非常大的改变才会变化主版本号。在一个库的生命周期中,为了引入一系列有用的特性,修改已存在的行为或者API等重大变更是在所难免的。因此,我们希望你能够爱上 Mockito 2.0!
重要变更 :
- Mockito从Hamcrest中解耦,自定义的matchers API也发生了改变,查看ArgumentMatcher 的基本原理以及迁移指南。
跟着我们的示例来mock 一个List,因为大家都知道它的接口(例如add(),get(), clear())。不要mock一个真实的List类型,使用一个真实的实例来替代。
0.1. Mockito支持Android
在2.6.1版本,我们提供了”原生的“Android支持。通过在项目中新增”mockito-android“库依赖,就能得到支持。这个工件发布到同一个Mockito组织,并且可以按照以下方式导入Android:
repositories {
jcenter()
}
dependencies {
testCompile "org.mockito:mockito-core:+"
androidTestCompile "org.mockito:mockito-android:+"
}
通过在“testCompile”范围中使用“mockito-core”工件,可以在常规VM上继续运行相同的单元测试,如上所示。请注意,由于Android VM的限制,您不能在Android上使用内联模拟生成器。如果您在Android上遇到模拟的问题,请在官方问题跟踪器上打开一个issue。一定要提供您正在使用的Android版本和项目的依赖项。
0.2. 免配置内联mock making
从2.7.6版本开始,我们提供”mockito-inline“ artifact,它支持内联的mock making而不需要配置MockMaker的扩自文件。使用方式,用”mockito-inline“替换”mockito-core“:
repositories {
jcenter()
}
dependencies {
testCompile "org.mockito:mockito-inline:+"
}
注意,这个artifact可能会停止维护,当内联mock making特性被集成到默认的mock maker里时。
查看更多关于内联mock making,查看39章
1. 验证某些行为
// 静态导入会使代码更简洁
import static org.mockito.Mockito.*;
// mock creation 创建mock对象
List mockedList = mock(List.class);
//using mock object 使用mock对象
mockedList.add("one");
mockedList.clear();
//verification 验证
verify(mockedList).add("one");
verify(mockedList).clear();
一旦mock对象被创建了,mock对象会记住所有的交互动作。然后你就可以选择性的验证你感兴趣的交互动作。
2. 如何做一些测试桩 (Stub)
// 你可以mock具体的类,不仅只是接口
LinkedList mockedList = mock(LinkedList.class);
// 测试桩
when(mockedList.get(0)).thenReturn("first");
when(mockedList.get(1)).thenThrow(new RuntimeException());
// 输出“first”
System.out.println(mockedList.get(0));
// 抛出异常
System.out.println(mockedList.get(1));
// 因为get(999) 没有打桩,因此输出null
System.out.println(mockedList.get(999));
//虽然能验证被打桩对象的调用情况,但这通常写起来很啰嗦
//如果你关注get(0)的返回值,那么其他地方就会中断(通常在verify()执行之前)
//如果你不关注get(0)的返回值,那么它就不应该被打桩。不相信?看下面例子。
verify(mockedList).get(0);
- 默认情况下,所有的函数都有返回值。mock函数会适当的返回null,原始类型/原始类型的包装类,一个空的集合,例如int/Integer返回0、boolean/Boolean返回false;
- 打桩动作可以被覆写 : 例如常见的打桩动作可以用于公共的配置,然后在测试函数中能够重新打桩。请注意,覆写测试桩函数是一种潜在的代码异味,它指出打桩动作太多了;
- 一旦测试桩函数被调用,该函数将会一直返回固定的值;
- 最后一个打桩动作是很重要的 - 当你用相同参数为一个方法做多次打桩时。换句话说:打桩的顺序很重要,但是这很少有意义,例如,当为完全相同的方法调用打桩,或者当使用参数匹配器时,等等。
3. 参数匹配器 (matchers)
Mockito以自然的java风格来验证参数值: 使用equals()函数。有时,当需要额外的灵活性时你可能需要使用参数匹配器,也就是argument matchers :
// 使用内置的anyInt()参数匹配器来打桩
when(mockedList.get(anyInt())).thenReturn("element");
// 使用自定义的参数匹配器来打桩( 在isValid()函数中返回你自己的匹配器实现 )
when(mockedList.contains(argThat(isValid()))).thenReturn("element");
// 输出“element”
System.out.println(mockedList.get(999));
// 你也可以用参数匹配器来验证
verify(mockedList).get(anyInt());
//参数匹配器也能用Java 8 Lambda风格编写
verify(mockedList).add(argThat(someString -> someString.length() > 5));
参数匹配器使验证和打桩变得更灵活。点击这里或这里查看更多内置的匹配器以及自定义参数匹配器或者hamcrest 匹配器的示例。
如果仅仅是获取自定义参数匹配器的信息,查看ArgumentMatcher类文档即可。
为了合理的使用复杂的参数匹配,使用原生风格的equals()匹配器,偶尔也用下anyX()匹配器,会使得测试代码更简洁、简单。有时,会迫使你重构代码以使用equals()匹配或者实现equals()函数来帮助你进行测试。
同时建议你阅读第15章节或者ArgumentCaptor类文档。ArgumentCaptor是一个能够捕获参数值的特殊参数匹配器。
参数匹配器的注意点 :
如果你使用了参数匹配器, 那么所有参数都要用匹配器。
示例 : ( 该示例展示了如何多次应用于测试桩函数的验证 )
verify(mock).someMethod(anyInt(), anyString(), eq("third argument"));
// 上述代码是正确的,因为eq()也是一个参数匹配器
verify(mock).someMethod(anyInt(), anyString(), "third argument");
// 上述代码是错误的,因为所有参数必须由匹配器提供,而参数"third argument"并非由参数匹配器提供,因此会抛出异常
像anyObject(), eq()这样的匹配器函数不会返回匹配器。它们会在内部将匹配器记录到一个栈当中,并且返回一个假的值,通常为null。这样的实现是由于被Java编译器强加的静态类型安全检查
。结果就是你不能在验证或者测试桩函数之外使用anyObject(), eq()函数。
4. 验证函数的确切、最少、从未调用的次数
//使用mock对象
mockedList.add("once");
mockedList.add("twice");
mockedList.add("twice");
mockedList.add("three times");
mockedList.add("three times");
mockedList.add("three times");
// 下面的两个验证函数效果一样,因为verify默认验证的就是times(1)
verify(mockedList).add("once");
verify(mockedList, times(1)).add("once");
// 验证具体的执行次数
verify(mockedList, times(2)).add("twice");
verify(mockedList, times(3)).add("three times");
// 使用never()进行验证,never相当于times(0)
verify(mockedList, never()).add("never happened");
// 使用atLeast()/atMost()进行验证
verify(mockedList, atLeastOnce()).add("three times");
verify(mockedList, atLeast(2)).add("five times");
verify(mockedList, atMost(5)).add("three times");
verify函数默认验证的是执行了times(1),也就是某个测试函数是否执行了1次.因此,times(1)通常被省略了。
5. 通过打桩为无返回值函数抛出异常
doThrow(new RuntimeException()).when(mockedList).clear();
// 调用这句代码会抛出运行时异常
mockedList.clear();
关于doThrow|doAnswer 等函数族的信息请阅读第十二章节。
6. 验证执行执行顺序
// A. 一个方法必须以指定顺序执行的模拟的对象
List singleMock = mock(List.class);
//使用这个模拟的对象
singleMock.add("was added first");
singleMock.add("was added second");
// 为该mock对象创建一个inOrder验证器
InOrder inOrder = inOrder(singleMock);
// 确保add函数首先执行的是add("was added first"),然后才是add("was added second")
inOrder.verify(singleMock).add("was added first");
inOrder.verify(singleMock).add("was added second");
// B .验证多个mock对象的必须以指定顺序被调用
List firstMock = mock(List.class);
List secondMock = mock(List.class);
//使用多个模拟的对象
firstMock.add("was called first");
secondMock.add("was called second");
// 创建inOrder对象,将需要验证的模拟对象传进来
InOrder inOrder = inOrder(firstMock, secondMock);
// 下面将会确认第一个模拟对象会在第二个模拟对象前被调用
inOrder.verify(firstMock).add("was called first");
inOrder.verify(secondMock).add("was called second");
// Oh, and A + B can be mixed together at will
验证执行顺序是非常灵活的 - 你不需要一个一个的验证所有交互,只需要验证你感兴趣的对象即可。
另外,你可以仅通过那些需要验证顺序的mock对象来创建InOrder对象。
7. 确认交互(interaction)操作没有在mock对象上执行
// 使用Mock对象 - 只有 mockOne发生了交互
mockOne.add("one");
// 普通验证
verify(mockOne).add("one");
// 验证一个模拟对象没有执行某个方法
verify(mockOne, never()).add("two");
// 验证其他mock对象没有交互
verifyZeroInteractions(mockTwo, mockThree);
8. 查找多余的调用
//using mocks
mockedList.add("one");
mockedList.add("two");
verify(mockedList).add("one");
// 下面的验证将会失败
verifyNoMoreInteractions(mockedList);
一些做了很多经典的、期望-执行-测试风格的模拟的用户,可能倾向于经常使用verifyNoMoreInteractions()
,甚至在每个测试函数中都用。但是verifyNoMoreInteractions()
并不建议在每个测试函数中都使用。verifyNoMoreInteractions()
在交互测试套件中只是一个便利的验证,它的作用是当你需要验证是否存在冗余调用时。滥用它将导致测试代码的可维护性降低。你可以阅读这篇文档来了解更多相关信息。
也可以看看never()
,它更加明确,并且很好地传达了意图。
9. 简化mock对象的创建 - @Mock注解
- 最小化重复的创建代码
- 使测试类的代码可读性更高
- 使验证错误更易于阅读,因为字段名可用于标识mock对象
public class ArticleManagerTest {
@Mock private ArticleCalculator calculator;
@Mock private ArticleDatabase database;
@Mock private UserProvider userProvider;
private ArticleManager manager;
注意! 下面这句代码需要在运行测试函数之前被调用,一般放到测试类的基类或者test runner中:
MockitoAnnotations.initMocks(testClass);
你可以使用内置的runner: MockitoJUnitRunner runner 或者一个rule : MockitoRule。关于JUnit5的测试案例,查看45章节关于JUnit扩展的描述。
关于mock注解的更多信息可以阅读MockitoAnnotations文档。
10. 为连续的调用打桩 (迭代器风格的打桩)
有时我们需要为同一个函数调用的不同的返回值/异常做测试桩。典型的运用案例就是对迭代器的模拟。
原始版本的Mockito并没有这个特性来促进模拟行为更简单,例如,可以使用Iterable或者简单的集合来替换迭代器。这些方法提供了更自然的方式,在一些场景中为连续的调用做测试桩会很有用。示例如下 :
when(mock.someMethod("some arg"))
.thenThrow(new RuntimeException())
.thenReturn("foo");
// 第一次调用 : 抛出运行时异常
mock.someMethod("some arg");
// 第二次调用 : 输出"foo"
System.out.println(mock.someMethod("some arg"));
// 后续调用 : 也是输出"foo"
System.out.println(mock.someMethod("some arg"));
另外,连续调用的另一种更简短的版本 :
// 第一次调用时返回"one",第二次返回"two",第三次返回"three"
when(mock.someMethod("some arg"))
.thenReturn("one", "two", "three");
注意:如果替代了“.thenReturn()"的调用,多个使用相同匹配器或参数的打桩动作,那么每个打桩动作都会覆盖之前的动作:
//All mock.someMethod("some arg") calls will return "two"
when(mock.someMethod("some arg"))
.thenReturn("one");
when(mock.someMethod("some arg"))
.thenReturn("two");
11. 通过回调方式来打桩
允许通过泛型Answer接口进行打桩。
在最初的Mockito里也没有这个具有争议性的特性。我们建议使用thenReturn() 或thenThrow()来打桩。这两种方法足够用于测试或者测试驱动开发。
如果你需要用泛型Answer接口来打桩,下面是个例子:
when(mock.someMethod(anyString())).thenAnswer(new Answer() {
Object answer(InvocationOnMock invocation) {
Object[] args = invocation.getArguments();
Object mock = invocation.getMock();
return "called with arguments: " + args;
}
});
// 输出 : "called with arguments: foo"
System.out.println(mock.someMethod("foo"));
12. doReturn()|doThrow()| doAnswer()|doNothing()|doCallRealMethod()系列方法
通过when(Object)
为无返回值的函数打桩有不同的方法,因为编译器不喜欢void函数在括号内…
当你想为void函数打桩来跑出一个异常时,使用doThrow()方法:
doThrow(new RuntimeException()).when(mockedList).clear();
// 下面的代码会抛出运行时异常
mockedList.clear();
当你调用doThrow()
, doAnswer()
, doNothing()
, doReturn()
and doCallRealMethod()
这些函数时,可以在适当的位置调用when()
函数. 当你需要下面这些功能时这是必须的:
- 测试void函数
- 在受监控的对象上测试函数
- 同一个函数多次打桩,为了在测试过程中改变mock对象的行为。
但是在调用when()
函数时你可以选择是否调用这些上述这些函数。
关于这些方法的信息:
- doReturn(Object)
- doThrow(Throwable)
- doThrow(Class)
- doAnswer(Answer)
- doNothing()
- doCallRealMethod()
13. 监控真实对象
你可以为真实对象创建多个监控(spy)对象。当使用这个spy对象时,真实的方法会被调用(除非它的函数被stub了)。尽量少使用spy对象,使用时也需要小心形式,例如spy对象可以用来处理遗留代码。
监控一个真实的对象可以与“局部mock对象”概念结合起来。在1.8之前,mockito的监控功能并不是真正的局部mock对象。原因是我们认为局部mock对象的实现方式并不好。在某些时候,我们发现了使用局部mock对象的合法用例。(第三方接口、临时重构遗留代码,完整的文章在这里 )
List list = new LinkedList();
List spy = spy(list);
//可选的,你可以为某些函数打桩:
when(spy.size()).thenReturn(100);
//通过spy对象调用真实对象的函数
spy.add("one");
spy.add("two");
//打印第一个元素"one"
System.out.println(spy.get(0));
//size()函数被打桩了,因此这里打印的是100
System.out.println(spy.size());
//可选的,你可以做校验
verify(spy).add("one");
verify(spy).add("two");
理解监控真实对象非常重要!
有时,在监控对象上使用when(Object)
来进行打桩是不可能或者不切实际的。因此,当使用监控对象时请考虑doReturn|Answer|Throw()
函数族来进行打桩。例如 :
List list = new LinkedList();
List spy = spy(list);
//不可能 : 当调用spy.get(0)时会调用真实对象的get(0)函数,此时会发生IndexOutOfBoundsException异常(因为真实list对象还是空的)
when(spy.get(0)).thenReturn("foo");
//你必须使用doReturn()来打桩
doReturn("foo").when(spy).get(0);
Mockito并不会代理传递来的真实对象,实际上它会拷贝一份真实对象。因此如果你保留了真实对象并且与之交互,不要期望从监控对象得到正确的结果。当你在监控对象上调用一个没有被stub的函数时,并不会调用真实对象的对应函数,你不会在真实对象上看到任何效果。
注意final方法,Mockito不能mock final方法,所以底线是:当你监视真实对象,你尝试在fianl方法上打桩,那么会有问题。同样,你也不能验证这些方法。
14. 修改没有测试桩的调用的默认返回值 ( 1.7版本之后 )
你可以创建mock对象,用指定策略来作为它的返回值。这是一个高级特性,通常来说,你不需要写这样的测试。无论如何,它对于遗留系统来说是很有用处的。
当你没有为函数调用打桩时,你可以指定一个默认的answer。
Foo mock = mock(Foo.class, Mockito.RETURNS_SMART_NULLS);
Foo mockTwo = mock(Foo.class, new YourOwnAnswer());
关于RETURNS_SMART_NULLS更多的信息请查看 :
RETURNS_SMART_NULLS文档 。
15. 为下一步的断言捕获参数 (1.8版本之后)
Mockito以java代码风格的形式来验证参数值 : 即通过使用equals()
函数。这也是我们推荐用于参数匹配的方式,因为这样会使得测试代码更简单、简洁。在某些情况下,当验证交互之后要检测真实的参数值时这将变得有用。例如 :
ArgumentCaptor<Person> argument = ArgumentCaptor.forClass(Person.class);
//参数捕获
verify(mock).doSomething(argument.capture());
//使用equal断言
assertEquals("John", argument.getValue().getName());
警告 : 我们建议在验证时使用ArgumentCaptor,而不是在打桩时。在打桩时使用ArgumentCaptor会降低测试代码的可读性,因为captor是在断言(又称为 验证/then)代码块之外创建的。另外,它可以降低本地化的缺点,因为如果测试桩函数没有被调用,那么参数就不会被捕获。
总之,ArgumentCaptor与自定义的参数匹配器相关(可以查看ArgumentMatcher类的文档 )。这两种技术都能用于检测外部传递到Mock对象的参数。然而,使用ArgumentCaptor在以下的情况下更合适 :
- 自定义不能被重用的参数匹配器
- 你仅需要断言参数值
打桩时用自定义参数匹配器更好点ArgumentMatcher。
16. 真实的局部模拟对象(mocks) (1.8版本之后)
在内部通过邮件进行了无数争辩和讨论后,最终 Mockito 决定支持部分mock,早前我们不支持是因为我们认为局部mocks会让代码变得糟糕。然而,我们发现了局部mocks真正合理的用法。详情点这
在 Mockito 1.8 之前,spy() 方法并不会产生真正的局部mocks,而这无疑会让一些开发者困惑。更详细的内容可以看:这里 或 Java 文档
//你能spy()方法创建局部模拟的对象:
List list = spy(new LinkedList());
//局部mock能力是可选的
Foo mock = mock(Foo.class);
//确保这个真实调用是'安全的'
//如果真实调用抛了异常或者依赖对象的特殊状态,那么就会有麻烦。(对象被mock,状态可能跟预期不一样。)
when(mock.someMethod()).thenCallRealMethod();
一如既往,你应该阅读局部mock的注意事项:面向对象编程通过把复杂系统拆分成独立个体来减少整体的复杂度,尤其是,SRPy对象。那局部mock是怎么遵循这个规范的呢?事实上它并没有遵循这个规范……局部mock通常意味着复杂性被移动到同一个对象的不同方法中,在大多数情况下,这不会是你想要的应用架构方式。
然而,在一些特殊的情况下局部mocks是方便的:处理不能轻易修改的代码(第三方接口,临时重构的遗留代码等)。然而,在新的测试驱动和架构优秀的代码上,我是不会使用局部mocks的。
17. 重置mocks对象 (1.8版本之后)
聪明的 Mockito 使用者很少会用到这个特性,因为他们知道这是出现糟糕测试单元的信号。通常情况下你不会需要重设你的mocks,只需要为每个测试方法重新创建一个mocks就可以了。
请考虑实现简单、小并且专注的测试方法,而不是冗长的、特殊的测试用例,来代替reset()。首先可能出现的代码异味的地方就是测试方法中间的 reset() 方法。这可能意味着你已经过度测试了。请倾听测试方法的声音:请让我们小巧的,而且专注在单一的行为上。在 Mockito 邮件列表中就有好几个和这个有关的讨论。
添加 reset() 方法的唯一原因是让它能与容器注入的mocks协作。详情看 issue 55 或 FAQ。
别自己给自己找麻烦,reset() 方法在测试方法中间确实是代码异味。
List mock = mock(List.class);
when(mock.size()).thenReturn(10);
mock.add(1);
reset(mock);
//在此刻,mock对象遗忘了所有的interactions & stubbing
18. 故障排查与验证框架的使用 (1.8版本之后)
首先,如果出现了任何问题,我建议你先看 Mockito FAQ。
万一有问题,你可以把问题提交到 Mockito 的邮件列表中。
然后,如果你一直正确的使用Mocktio,你应该知道Mockito的检查机制。对此有疑惑的话不妨看看 validateMockitoUsage() 的文档说明。
19.行为驱动开发的别名 (1.8版本之后)
用行为驱动开发的风格来写测试用例,会使用//given //when //then 注解作为测试方法的基础部分。这正是我们编写测试的方式,我们热情地鼓励您这样做!
你可以在这开始学习有关 BDD 的知识
问题是:当前打桩API中,‘when’的规范角色没有很好的集成到//given //when //then注释中。这是因为测试打桩属于测试中的given部分,而且不是测试中的when部分。因此 BDDMockito 类介绍了一个别名,使你的测试桩方法调用 BDDMockito.given(Object) 方法。现在它可以很好地和BDD风格的测试中的given部分进行交互。
测试用例看起来会是这样子:
import static org.mockito.BDDMockito.*;
Seller seller = mock(Seller.class);
Shop shop = new Shop(seller);
public void shouldBuyBread() throws Exception {
//given
given(seller.askForBread()).willReturn(new Bread());
//when
Goods goods = shop.buyBread();
//then
assertThat(goods, containBread());
}
20. 可序列化的mocks(1.8.1版本之后)
模拟的对象是可以被序列化的。有了这个特性,你就可以在要求依赖项是可序列化的地方,使用模拟对象。
警告:这个特性应该在单元测试中少用。
实现这个特性是为了DBB规范的一个特例:有一个不可靠的外部依赖。在web环境下,来自外部依赖的对象为了在不同层之间传递,会被序列化。
用MockSettings.serializable():创建一个可序列化的mock对象:
List serializableMock = mock(List.class, withSettings().serializable());
假设所有普通的序列化要求都被类满足了,模拟的对象是能被序列化的。
让一个真实的侦查对象可序列化需要多一些努力,因为 spy(…) 方法没有接收 MockSettings 参数的重载版本。不过不用担心,你几乎不可能用到这。
List<Object> list = new ArrayList<Object>();
List<Object> spy = mock(ArrayList.class, withSettings()
.spiedInstance(list)
.defaultAnswer(CALLS_REAL_METHODS)
.serializable());
21. 新的注解 : @Captor,@Spy,@InjectMocks (1.8.3版本之后)
V1.8.3 带来的新注解在某些场景下可能会很实用:
@Captor 简化 ArgumentCaptor 的创建 - 当需要捕获的参数是一个令人讨厌的泛型类,而且你想避免编译时警告。
@Spy - 你可以用它代替 spy(Object) 方法
@InjectMocks - 自动将模拟或监视的对象注入到被测试对象中。
需要注意的是 @InjectMocks 也能与 @Spy 一起使用,这就意味着 Mockito会在测试中,将mocks对象注入局部mock对象中。这变得很复杂,所以你还是应该少用局部mock。参考16点关于局部mock的介绍。
所有新的注解都是只在MockitoAnnotations.initMocks(Object)被处理。就像@Mock注解,你能用内置runner(MockitoJUnitRunner 或 规则: MockitoRule)来开启。
所有新的注解都是只在[MockitoAnnotations.initMocks(Object)]
22. 带超时的验证 (1.8.5版本之后)
允许带超时的验证。这使得一个验证会等待一段特定的时间,以获得想要的交互,而不是还没有发生事件就立即失败(即超时时间到了才会失败)。在并发条件下的测试这会很有用。
这个特性应该少被使用 - 找到更好的方法去测试多线程系统。
还没有实现和 InOrder 验证协作。
例子:
//someMethod()执行时间不少于100ms才会结束
//校验通过时马上会马上结束(可能不会等到100ms)
verify(mock, timeout(100)).someMethod();
//上一行例子是下面这行代码的别名:
verify(mock, timeout(100).times(1)).someMethod();
//someMethod()在100ms内执行了2次就会马上通过
verify(mock, timeout(100).times(2)).someMethod();
//等效: someMethod()在100ms内执行了2次就会马上通过
verify(mock, timeout(100).atLeast(2)).someMethod();
23. 自动实例化被@Spies, @InjectMocks注释的字段以及构造函数注入 (1.9.0版本之后)
Mockito 现在会通过构造器注入、setter注入 或字段注入方式,尽可能初始化带有 @Spy 和 @InjectMocks 注解的字段。
为了利用这个特性,你需要使用 MockitoAnnotations.initMocks(Object), MockitoJUnitRunner 或 MockitoRule。
请在InjectMocks的javadoc查看更多的使用技巧和注入规范
//旧的写法:
@Spy BeerDrinker drinker = new BeerDrinker();
//新的写法:
@Spy BeerDrinker drinker;
//@InjectMocks注解也可以这么用:
@InjectMocks LocalPub;
24. 单行测试桩 (1.9.0版本之后)
Mockito 现在允许在打桩时创建模拟对象。主要是,它允许通过一行代码来创建一个测试桩。这对保持代码的整洁很有用。举例来说,在测试字段初始化时,有些简单的测试桩可以被创建并赋值到字段上,例如:
public class CarTest {
Car boringStubbedCar = when(mock(Car.class).shiftGear()).thenThrow(EngineNotStarted.class).getMock();
@Test public void should... {}
25. 验证被忽略的测试桩 (1.9.0版本之后)
Mockito 现在允许为了更好的验证而忽略已有的测试桩。与 verifyNoMoreInteractions() 方法或验证 inOrder() 方法组合使用,有时很有用。帮助避免多余的测试桩调用校验 - 显然我们不会对验证测试桩感兴趣。
警告,ignoreStubs() 可能会导致 verifyNoMoreInteractions(ignoreStubs(…)) 方法的过度使用。谨记在心,Mockito 没有推荐在每一个测试中用 verifyNoMoreInteractions() 方法,原因在 verifyNoMoreInteractions(Object…)的Java 文档中有。
一些例子:
verify(mock).foo();
verify(mockTwo).bar();
//忽略所有已经打桩的方法:
verifyNoMoreInvocations(ignoreStubs(mock, mockTwo));
//创建 InOrder 将会忽略打桩信息
InOrder inOrder = inOrder(ignoreStubs(mock, mockTwo));
inOrder.verify(mock).foo();
inOrder.verify(mockTwo).bar();
inOrder.verifyNoMoreInteractions();
更好的例子和更多的细节都可以在 Java 文档的 ignoreStubs(Object…) 部分看到。
26. mock详情 (2.2.x中改进)
Mockito 提供API来查看mock对象的详情。这些API对高级用户和mock框架整合者很有用。
//识别特定的对象是一个mock对象或者是spy对象:
Mockito.mockingDetails(someObject).isMock();
Mockito.mockingDetails(someObject).isSpy();
//获取mock对象的详情信息,包括它的类型或默认的answer:
MockingDetails details = mockingDetails(mock);
details.getMockCreationSettings().getTypeToMock();
details.getMockCreationSettings().getDefaultAnswer();
//获取mock对象的执行器(invocations)和打桩信息(stubbings):
MockingDetails details = mockingDetails(mock);
details.getInvocations();
details.getStubbings();
//打印所有的交互(including stubbing, unused stubs)
System.out.println(mockingDetails(mock).printInvocations());
更多信息请查看MockingDetails的java文档.
27. 真实实例的委托调用 (Since 1.9.5)
代理调用对间谍或局部mock的对象有用的,这些对象用常规 spy API很难去mock或spy。从 Mockito 的 1.10.11 版本开始, delegate 有可能和 mock 的类型相同也可能不同。如果不是同一类型,delegate 类型需要提供一个匹配方法否则就会抛出一个异常。可能用到这个特性的案例:
- 带有 interface 的 final 类
- 已经自定义代理的对象
- 带有 finalize 方法的特殊对象,就是避免执行2次。
和常规 spy 的不同:
-
标准的 spy (spy(Object)) 包含被监视实例的所有状态信息,方法在 spy 对象上被调用。被监视的对象只在 mock
创建时被用来拷贝状态信息。如果你通过标准 spy 调用一个方法,这个 spy 会调用其内部的其他方法记录这次操作,
以便后面验证使用。等效于存根 (stubbed)操作。 -
代理方式的 mock 只是简单的把所有方法委托给 delegate。delegate 一直被当成它所代理的方法使用。如果你调用委托的mock对象上的方法,它会调用其内部的其他方法,这些调用不会被记录,打桩动作 在这里也不会生效。委托方式的Mock 相对于标准的 spy 来说功能弱了很多,不过在标准 spy 不能被创建的时候这还是很有用。
更多信息可以看这里 AdditionalAnswers.delegatesTo(Object).
28. MockMaker API (Since 1.9.5)
为了满足用谷歌Android用户的需求,Mockito 现在提供一个扩展点,允许替换代理生成引擎。默认情况下,Mockito 使用 Byte Buddy 创建动态代理。
这个扩展点是为想要扩展 Mockito 功能的高级用户准备的。比如,我们现在就可以在 dexmaker 的帮助下使用 Mockito
测试 Android。
更多的细节,原因和示例请看 MockMaker 的文档。
29. BDD 风格的验证 (Since 1.10.0)
开始验证时,使用then关键字可以开启 Behavior Driven Development (BDD) 风格的验证。
given(dog.bark()).willReturn(2);
// when
...
then(person).should(times(2)).ride(bike);
更多信息请查阅 BDDMockito.then(Object) .
30. 监视 或 模拟 抽象类 (1.10.12版本加入,在2.7.13 和 2.7.14版特征得到增强))
现在可以方便的 spy 一个抽象类。注意,过度使用 spy 或许意味着代码的设计上有问题。(see spy(Object)).
之前,spying 只可以用在实例对象上。而现在新的 API 可以使用构造函数来创建一个 mock 实例。这对 mock
一个抽象类来说是很重要的,这样使用者就不必再提供一个抽象类的实例了。目前的话只支持无参构造函数,
如果你认为这样还不够的话欢迎向我们反馈。
//方便的API,新增重载的spy()方法:
SomeAbstract spy = spy(SomeAbstract.class);
//模拟抽象方法,监视接口默认方法(只有在2.7.13后才可用)
Function function = spy(Function.class);
//健壮的 API,通过配置构造器:
OtherAbstract spy = mock(OtherAbstract.class, withSettings()
.useConstructor().defaultAnswer(CALLS_REAL_METHODS));
//模拟构造器带参数的抽象类(只有在2.7.14后才可用)
SomeAbstract spy = mock(SomeAbstract.class, withSettings()
.useConstructor("arg1", 123).defaultAnswer(CALLS_REAL_METHODS));
//模拟非静态内部抽象类:
InnerAbstract spy = mock(InnerAbstract.class, withSettings()
.useConstructor().outerInstance(outerInstance).defaultAnswer(CALLS_REAL_METHODS));
更多信息请见 MockSettings.useConstructor() .
31. Mockito的模拟对象 可以通过 classloaders 序列化/反序列化 (Since 1.10.0)
Mockito 通过 classloader 引入序列化。和其他形式的序列化一样,所有 mock 层的对象类型都要可序列化,
包括 answers。因为序列化模式需要大量的工作,所以这是一个可选配置。
// 常规的 serialization
mock(Book.class, withSettings().serializable());
// 通过 classloaders 序列化
mock(Book.class, withSettings().serializable(ACROSS_CLASSLOADERS));
更多信息请查看 MockSettings.serializable(SerializableMode).
32. Deep stubs 更好的泛型支持 (Since 1.10.0)
深度打桩模式 现在可以更好的查找类的泛型信息。这就意味着像这样的类
不必去 mock 它的行为就可以使用。
class Lines extends List<Line> {
// ...
}
lines = mock(Lines.class, RETURNS_DEEP_STUBS);
//现在 Mockito 知道这是Line类型,而不是Object
Line line = lines.iterator().next();
请注意,大多数情况下 mock 返回一个 mock 对象是错误的。
33. Mockito JUnit rule (Since 1.10.17)
Mockito 现在提供一个 JUnit 的 rule。目前为止,有两种方法可以初始化 fields ,这些fields使用了 Mockito 提供的注解比如
@Mock, @Spy, @InjectMocks 等等。
- 用 @RunWith(@MockitoJUnitRunner.class) 标注 JUnit 测试类
- 在 @Before 方法中调用 MockitoAnnotations.initMocks(Object)
现在你可以选择使用一个 rule:
@RunWith(YetAnotherRunner.class)
public class TheTest {
@Rule public MockitoRule mockito = MockitoJUnit.rule();
// ...
}
更多信息到这里查看 MockitoJUnit.rule().
34. 开启和关闭插件的开关 (Since 1.10.15)
这是一个测试特性,可以控制一个 mockito-plugin 开启或者关闭。详情请查看 PluginSwitch
35. 自定义验证失败信息 (Since 2.1.0)
允许声明一个在验证失败时输出的自定义消息
示例:
// 在验证失败时,会打印自定义的消息
verify(mock, description("This will print on failure")).someMethod();
// 任何验证模式下都能使用这种方式
verify(mock, times(2).description("someMethod should be called twice")).someMethod();
36. Java 8 Lambda匹配器的支持 (Since 2.1.0)
你可以在参数匹配器(ArgumentMatcher)上使用Java 8 lambda表达式,来减少对参数捕获器(ArgumentCaptor)的依赖。如果你需要验证mock对象上方法调用的输入是正确的,那么你需要正常使用参数捕获器来找到使用过的操作数,并且之后对它们做断言。对于复杂的例子这是有用的,当然它也很啰嗦。
写一个lambda来表示匹配关系是很容易的。你方法的参数使用argThat进行连接时,参数将作为强类型对象传给参数匹配器,所以通过这种方式能做任何事情。
例如:
// 验证一个list只添加了某些长度的字符串(字符串长度小于5)
// 注意 - 这种写法只有在Java 8下能编译通过
verify(list, times(2)).add(argThat(string -> string.length() < 5));
// Java 7 等效的 - 不太整洁
verify(list, times(2)).add(argThat(new ArgumentMatcher(){
public boolean matches(String arg) {
return arg.length() < 5;
}
}));
// Java 8 下,更复杂的例子 - 你可以通过函数式指定复杂的验证行为
verify(target, times(1)).receiveComplexObject(argThat(obj -> obj.getSubObject().get(0).equals("expected")));
// lambda的方式也可以被用到:定义不同入参下,mock对象的行为
// in this case if the input list was fewer than 3 items the mock returns null
when(mock.someMethod(argThat(list -> list.size()<3))).thenReturn(null);
37. Java 8 自定义Answer的支持 (Since 2.1.0)
Answer接口只有一个方法,Java 8 使用lambda表达式来实现它非常简单。你越需要使用方法调用的参数,就越需要对来自InvocationOnMock的参数进行类型转换。
例如:
// answer每次都返回12
doAnswer(invocation -> 12).when(mock).doSomething();
// 用参数里的一个值作为返回值 - 转换成你想要的正确类型
//- 在这个例子里,把第二个字符串类型的参数的长度,作为返回值。
//随着参数的增长,这很快就会变得冗长。
doAnswer(invocation -> ((String)invocation.getArgument(1)).length())
.when(mock).doSomething(anyString(), anyString(), anyString());
方便起见,现在我们可以定义这样的answer/actions,用被调用方法的参数作为answer/actions中lambda的入参。即使在Java 7和更低版本,自定义基于类型化接口的answers也能减少样板代码。在特定场景,这种方法使测试使用了回调的方法更加容易。answer() 和 answerVoid()方法可以创建answer对象。它们依赖的相关answer接口在org.mockito.stubbing包下,Answer接口最多支持5个参数。
例如:
//例如,将要被模拟的接口有一个这样的方法:
void execute(String operand, Callback callback);
//这个例子的回调函数有一个方法,测试类会依赖这个回调的执行
void receive(String item);
// Java 8 - style 1
doAnswer(AdditionalAnswers.answerVoid((operand, callback) -> callback.receive("dummy"))
.when(mock).execute(anyString(), any(Callback.class));
// Java 8 - style 2 - 假设AdditionAnswers已经静态导入
doAnswer(answerVoid((String operand, Callback callback) -> callback.receive("dummy"))
.when(mock).execute(anyString(), any(Callback.class));
// Java 8 - style 3 - 被模拟的方法是测试类的静态成员
private static void dummyCallbackImpl(String operation, Callback callback) {
callback.receive("dummy");
}
doAnswer(answerVoid(TestClass::dummyCallbackImpl)
.when(mock).execute(anyString(), any(Callback.class));
// Java 7
doAnswer(answerVoid(new VoidAnswer2() {
public void answer(String operation, Callback callback) {
callback.receive("dummy");
}})).when(mock).execute(anyString(), any(Callback.class));
// 用answer() 方法返回一个值是可能的,
// 并且有返回值版本的函数式接口也是如此,
// 如果模拟的接口有一个这样的方法:
boolean isSameString(String input1, String input2);
// Java 8 - 这种方式是可以被模拟的
doAnswer(AdditionalAnswers.answer((input1, input2) -> input1.equals(input2))))
.when(mock).execute(anyString(), anyString());
// Java 7
doAnswer(answer(new Answer2() {
public String answer(String input1, String input2) {
return input1 + input2;
}})).when(mock).execute(anyString(), anyString());
38. 元数据和泛型信息保留 (Since 2.1.0)
Mockito 现在会保留mock方法和类上的注解信息,也会保留泛型的元信息。以前,mock类型不保存类上的注解信息除非注解被显示地继承,并且不保存方法上的注解信息。因此,下面的情况现在是有效的:
@MyAnnotation
class Foo {
List<String> bar() { ... }
}
Class<?> mockType = mock(Foo.class).getClass();
assert mockType.isAnnotationPresent(MyAnnotation.class);
assert mockType.getDeclaredMethod("bar").getGenericReturnType() instanceof ParameterizedType;
使用 Java 8,Mockito 现在也保存类型注解(type annotations)。这是默认行为,如果有可选择的MockMaker被使用,这种行为可能不会持续。
39. 模拟final类型,枚举 和 final方法 (Since 2.1.0)
Mockito 现在对模拟final类和方法提供一个孵化中的,可选的的支持。这是一个极好的改进,说明了Mockito永恒的追求:提高测试体验。我们的志向是:Mockito要支持final类和方法。以前,它们被认为是不可模拟的,阻止用户去模拟的。我们已经开始讨论怎么使这个特性被默认支持。当前,这个特性仍然是可选的,我们等待社区里更多的反馈。
这个可选的mock maker,它使用Java instrumentation API和子类两者结合的方式,而不是创建一个新的类来代表这个mock对象。通过这种方式,使模拟final类和方法成为了可能。
这个mock maker默认是关闭的,因为它是基于完全不同的模拟机制,这种机制需要社区更多的反馈。它能通过mockito的扩展机制来被激活,只要在classpath上创建一个文件/mockito-extensions/org.mockito.plugins.MockMaker,文件包含mock-maker-inline这个值。
方便起见,Mockito团队提供了一个artifact,这个artifact里mock maker已经配置好。在项目中引入mockito-inline artifact来代替mockito-core artifact。注意,等到对final类和方法的模拟功能集成到默认的mock maker后,这个artifact可能会停止。
关于这个mock maker几个需要注意的点是:
- 模拟final类和枚举的mock配置是不兼容的:
- 显示的序列化支持 withSettings().serializable()
- 特殊接口 withSettings().extraInterfaces()
- 一些方法不能被模拟
- java.*包下的方法
- 原生方法
- 这个mock maker被设计成围绕着Java Agent运行时的附件(attachment);这个要求一个兼容的JVM,JVM是JDK的一部分(或Java 9的虚拟机)。当运行在Java 9之前没JDK的虚拟机上,可以在JVM启动时,通过-javaagent参数,手动添加Byte Buddy的jar包。
如果你对这个特性的细节感兴趣,请阅读org.mockito.internal.creation.bytebuddy.InlineByteBuddyMockMaker的java文档。
40. “严格的”Mocktio能提高生产效率并使测试用例更清晰(2.+版本之后)
为什么“严格的”Mockito能使你的生产率更好,并使你的测试更整洁,看:
- 通过JUnit Rules来开启严格打桩 - MockitoRule.strictness(Strictness) with Strictness.STRICT_STUBS
- 通过Junit Runner来开启严格打桩 - MockitoJUnitRunner.StrictStubs
- 如果你不能使用runner/rule(例如使用TestNG)来开启严格打桩 - MockitoSession
- 通过MockitoJUnitRunner发现非必要测试桩
- 打桩参数不匹配的警告,记录在MockitoHint中
Mockito默认是一个“宽松的”模拟框架。mock对象可以做交互,而不用事设置期望的行为。这是有意设计的,它通过强制用户明确地表达他们想要 打桩/验证的内容来提高测试的质量。它也非常直观的,易于使用,并且和“given”,“when”,“then”整洁的测试代码模板完美融合在一起。这也是区别于过去其他的经典模拟框架,过去的框架默认都是严格的。
默认是“宽松的”,有时让Mockito的测试用例难于被debug。有些情况下,配置错误的打桩强制用户使用调试器执行测试用例。理想状况下,测试用例失败是显而易见的结果,并且不需要调试器来定位根本原因。从2.1版本开始,Mockito拥有一些新的特性,这些特性推动框架走向“严格性”。我们想让Mockito提供出色的调试能力,同时也不丢失它核心的模拟风格,针对直观性,清晰性和整洁的测试代码进行优化。
帮助 Mockito!尝试这些新的特性,给我们反馈,加入GitHub issue 769关于Mockito严格性的讨论。
41. 框架集成的高级公开API (2.10.+版本之后)
在2017的夏天,我们觉得Mockito应该为高级框架集成提供更好的API。这个新的API不是为了写单元测试的用户。它旨在用于需要用一些定制逻辑扩展或包装Mockito的其他测试工具和模拟框架。在设计期间和实现过程中(issue 1110),我们开发和改变了下列公开API的元素:
- 新的 MockitoPlugins - 使框架集成者能够访问默认的Mockito插件。当需要实现自定义插件(例如MockMaker),并且将某些行为委托给默认Mockito的实现。
- 新的 MockSettings.build(Class) - 创建模拟配置的不可变的视图,供Mockito稍后使用。使用InvocationFactory创建invocations 或 当实现自定义MockHandler时,视图将非常有用。
- 新的 MockingDetails.getMockHandler() - 其他框架可能使用这个mock处理器,以编程方式对mock对象模拟调用。
- 新的 MockHandler.getMockSettings() - 用于获取创建模拟对象的配置。
- 新的 InvocationFactory - 提供创建调用对象实例的方法。有益于需要以编程方式模拟在mock对象上的方法调用的框架集成。
- 新的 MockHandler.getInvocationContainer() - 提供获取没有方法的调用容器对象(makrer接口)。容器需要隐藏它内部的实现机制,并且避免将其泄露给公共API。
- 改变Stubbing接口 - Stubbing现在继承自 Answer 接口。它是向后兼容的,因为Stubbing接口不是一个可扩展的(看@NotExtensible注解)。这个改变对用户来说是无感的。
- InternalMockHandler标记为过期的 - 为了适应API变化,我们需要标记这个接口成过期的。这个接口一直记录为内部的,我们没有证据证明它被社区使用。这个弃用行为应该对用户完成无感。
- NotExtensible - 公共的注释,指示用户不应该对提供给定类型的自定义实现。帮助框架集成者和我们的用户理解怎么去安全的使用Mockito API。
你有反馈吗?请在issue 1110里留言。
42. 集成新的API: 监听验证开始(verification start)事件(2.11.+版本之后)
Spring Boot等框架集成需要公共API来处理双代理用例(issue 1191)。我们新增的:
- 新的 VerificationStartedListener 和 VerificationStartedEvent 使框架集成者替换模拟对象以进行验证。主要的驱动用例是Spring Boot的集成。详情请看VerificationStartedListener的java文档。
- 新的公共方法MockSettings.verificationStartedListeners(VerificationStartedListener…)允许在mock创建时,提供验证启动的监听器。
- 新的方便的方法MockingDetails.getMock()被添加,用来使MockingDetails API更加完整。我们发现这个方法在实例化时非常有用。
43. 集成新的API: 测试框架支持MockitoSession(2.15.+版本之后)
MockitoSessionBuilder and MockitoSession得到增强,以便通过测试框架集成实现重用(例如MockitoRule对于JUnit一样)。
- MockitoSessionBuilder.initMocks(Object…) 允许传入多个测试类的实例,用来初始化被Mockito注解(例如@Mock)注释的字段。当测试使用多个(例如内嵌的)测试类实例时,这个方法对高级框架集成非常有用(例如 JUnit Jupiter)。
- MockitoSessionBuilder.name(String)允许将名称从测试框架传递到MockitoSession,当使用Strictness.WARN时,MockitoSession将用于打印警告。
- MockitoSessionBuilder.logger(MockitoSessionLogger)使定制,用于完成模拟时产生的提示/告警 的logger成为了可能(对于测试和连接JUnit Jupiter等测试框架提供的报告功能很有用)。
- MockitoSession.setStrictness(Strictness)允许在一次性情况下,改变一个MockitoSession的严格性。例如,它可以为一个类下的所有测试用例设置一个默认的严格性,但是也可以改变单个或几个测试用例的严格性。
- 添加MockitoSession.finishMocking(Throwable)用于避免因多个竞争的失败而可能出现的混乱。当提供的失败用例不是null时,它会关闭某些检查。
44. org.mockito.plugins.InstantiatorProvider泄露内部API所以被org.mockito.plugins.InstantiatorProvider2替代(2.15.4版本之后)
InstantiatorProvider返回了一个内部API。因此它被启用并且被[InstantiatorProvider2][InstantiatorProvider2]替代。旧的调用者提供商将继续工作,但是它被推荐切换到新的API。
[InstantiatorProvider2https://static.javadoc.io/org.mockito/mockito-core/2.23.4/org/mockito/plugins/InstantiatorProvider2.html
45. JUnit5+的扩展
想要和JUnit Jupiter(JUnit5+)集成,可以使用org.mockito:mockito-junit-jupiter
artifact。有关集成使用的更多信息,请查阅MockitoExtension的Java文档。
46. 新的Mockito.lenient()和MockSettings.lenient()方法(2.20.0版本之后)
在Mockito2早期开始,就有严格打桩的特性。它非常有用,因为它驱动更整洁的测试用例和提供生产率。严格打桩上报非必要的桩点,检查打桩参数不匹配 并且使测试用例更DRY(Strictness.STRICT_STUBS)。这需要权衡:在某些情况,你可能从严格打桩得到错误的否定。为了补救这种情况,你可以将特定打桩配置为宽松的,而所有其他的打桩和mocks使用严格打桩:
lenient().when(mock.foo()).thenReturn("ok");
如果你想在给定的mock对象的所有打桩都是宽松的,你可以按照下面的方式配置mock对象:
Foo mock = Mockito.mock(Foo.class, withSettings().lenient());
查阅lenient()更多的信息。开启一个GitHub的issue来讨论,让我们知道你是怎么发现这个新特性。
字段摘要
类型 | 字段以及描述 |
static Answer< Object > | CALLS_REAL_METHODS 用于mock(Class, Answer)的可选参数Answer |
字段详情
**CALLS_REAL_METHODS** |
public static final Answer CALLS_REAL_METHODS 用于mock(Class, Answer)的可选参数Answer Answer可以用于定义unstubbed invocations的返回值. |
这个Answer接口对于legacy code非常有用. 当使用这个接口的时候, unstubbed methods会被实现.
这是一种通过调用默认方法来创建partial mock对象的方式。
通常,你将要阅读mock的部分警告:Object oriented programming is more less tackling complexity by dividing the complexity into separate, specific, , SRPy objects.
partial mock是如果适应这种模式的呢?好吧!它不仅仅是,partial mock通常意味着复杂性在同一个对象中移动到不同的方法.在大多数情况下,这不是你想要的设计你的应用的方式。
然而,当partial mocks派上用场同样也有少许情况:处理你不易改变的代码(第三方接口,legacy code的临时重构).我将不使用partial mocks用于新的、测试驱动以及设计不错的代码。
例如:
Foo mock = mock(Foo.class, CALLS_REAL_METHODS); // this calls the real implementation of Foo.getSomething() value = mock.getSomething(); when(mock.getSomething()).thenReturn(fakeValue); // now fakeValue is returned value = mock.getSomething(); |
方法摘要
Modifier and Type | Method and Description |
static VerificationAfterDelay | after(long millis) 给定的时间后进行验证 |
static VerificationMode | atLeast(int minNumberOfInvocations) 至少进行minNumberOfInvocations次验证 |
static VerificationMode | atLeastOnce() 至少进行一次验证 |
static VerificationMode | atMost(int maxNumberOfInvocations) 最多进行maxNumberOfInvocations次验证 |
static VerificationMode | calls(int wantedNumberOfInvocations) 允许顺序进行non-greedy验证 |
方法详情
after
public static VerificationAfterDelay after(long millis)
在给定的时间后进行验证。它会为了预期的效果进行等待一段时间后进行验证,而不是因为没发生而立即失败。这可能对于测试多并发条件非常有用。
after()等待整个周期的特点不同于timeout(),而timeout()一旦验证通过就尽快停止,例如:当使用times(2)可以产生不同的行为方式,可能通过后过会又失败。这种情况下,timeout只要times(2)通过就会通过,然后after执行完整个周期时间,可能会失败,也意味着times(2)也失败。
感觉这个方法应该少使用——找到更好的方法测试你的多线程系统。
对尚未实现的工作进行验证。
//passes after 100ms, if someMethod() has only been called once at that time.<br>
verify(mock, after(100)).someMethod();<br>
//above is an alias to:<br>
verify(mock, after(100).times(1)).someMethod();
//passes if someMethod() is called *exactly* 2 times after the given timespan
verify(mock, after(100).times(2)).someMethod();
//passes if someMethod() has not been called after the given timespan<br>
verify(mock, after(100).never()).someMethod();
//verifies someMethod() after a given time span using given verification mode
//useful only if you have your own custom verification modes.
verify(mock, new After(100, yourOwnVerificationMode)).someMethod();
参照Mockito类的javadoc帮助文档中的例子
Parameters:
- millis - - time span in milliseconds
Returns:
- verification mode
atLeast
public static VerificationMode atLeast(int minNumberOfInvocations)
允许至少进行x次验证。例如:
verify(mock, atLeast(3)).someMethod("some arg");
参照Mockito类的javadoc帮助文档中的例子
Parameters:
- minNumberOfInvocations - invocations的最小次数
Returns:
- verification mode
atLeastOnce
public static VerificationMode atLeastOnce()
至少进行一次一次验证。例如:
verify(mock, atLeastOnce()).someMethod("some arg");
atLeast(1)的别名.
参照Mockito类的javadoc帮助文档中的例子
Returns:
- verification mode
atMost
public static VerificationMode atMost(int maxNumberOfInvocations)
至多进行x次验证. 例如:
verify(mock, atMost(3)).someMethod("some arg");
参照Mockito类的javadoc帮助文档中的例子
Parameters::
- maxNumberOfInvocations - invocations的最大次数
Returns:
- verification mode
calls
public static VerificationMode calls(int wantedNumberOfInvocations)
允许顺序进行non-greedy验证. 例如:
inOrder.verify( mock, calls( 2 )).someMethod( "some arg" );
- 如果这个方法调用3次不会失败,不同于times(2)
- 不会标记第三次验证,不同于atLeast(2)
Parameters::
- wantedNumberOfInvocations - 验证的次数
Returns:
- verification mode
继承org.mockito.Matchers的方法
any
public static <T> T any()<br><br>
匹配任何值,包括null
anyObject()的别名
参照Matchers类的javadoc帮助文档中的例子
这是: anyObject() and any(java.lang.Class)的别名
Returns:
- null
any
public static <T> T any(Class<T> clazz)
匹配任何对象,包括null
这个方法不进行给定参数的类型检查,它就是为了避免代码中的强制转换(Casting)。然而这可能会改变(类型检查可以添加)将来的主要版本。
参照Matchers类的javadoc帮助文档中的例子
这是: anyObject() and any(java.lang.Class)的别名
Returns:
- null
anyBoolean
public static boolean anyBoolean()
任何boolean类型或非空(non-null)的Boolean.
参照Matchers类的javadoc帮助文档中的例子
Returns:
- false
anyByte
public static byte anyByte()
任何byte类型变量或非空(non-null)Byte.
参照Matchers类的javadoc帮助文档中的例子
Returns:
- 0
anyChar
public static char anyChar()
任何char类型变量或非空(non-null)的Character.
参照Matchers类的javadoc帮助文档中的例子
Returns:
- 0
anyCollection
public static Collection anyCollection()
任何非空(non-null)的Collection.
参照Matchers类的javadoc帮助文档中的例子
Returns:
- 空Collection.
anyCollectionOf
public static < T > Collection < T > anyCollectionOf(Class<T> clazz)
通用友好的别名anyCollection()。为了保持代码清洁,通过@SuppressWarnings(“unchecked”)来进行替代编译器警告。
任何非空(non-null)Collection.
这个方法不进行给定参数的类型检查,它就是为了避免代码中的强制转换(Casting)。然而这可能会改变(类型检查可以添加)将来的主要版本。
参照Matchers类的javadoc帮助文档中的例子
Parameters
- clazz - 类型属于Collection类型避免类型转换(Casting)
Returns:
- 空Collection.
anyDouble
public static double anyDouble()
任何double类型或非空(non-null)的Double.
参照Matchers类的javadoc帮助文档中的例子
Returns:
anyFloat
public static float anyFloat()
任何float类型或非空(non-null)Float.
参照Matchers类的javadoc帮助文档中的例子
Returns:
anyInt
public static int anyInt()
任何int或非空(non-null)Integer.
参照Matchers类的javadoc帮助文档中的例子
Returns:
anyList
public static List anyList()
任何非空(non-null)List.
参照Matchers类的javadoc帮助文档中的例子
Returns:
- 空List.
anyListOf
public static < T > List < T > anyListOf(Class< T > clazz)
通用友好的别名anyList()。为了保持代码清洁,通过@SuppressWarnings(“unchecked”)来进行替代编译器警告。
任何非空(non-null)List.
这个方法不进行给定参数的类型检查,它就是为了避免代码中的强制转换(Casting)。然而这可能会改变(类型检查可以添加)将来的主要版本。
参照Matchers类的javadoc帮助文档中的例子
Parameters:
- clazz - 类型属于List类型避免类型转换(Casting)
Returns:
- 空List.
anyLong
public static long anyLong()
任何long类型或非空(non-null)Long.
参照Matchers类的javadoc帮助文档中的例子
Returns:
anyMap
public static Map anyMap()
任何非空(non-null)Map.
参照Matchers类的javadoc帮助文档中的例子
Returns:
- 空Map.
anyMapOf
public static < K,V> Map < K,V> anyMapOf(Class< K> keyClazz, Class< V> valueClazz)
通用友好的别名anyMap()。为了保持代码清洁,通过@SuppressWarnings(“unchecked”)来进行替代编译器警告。
任何非空(non-null)Map.
这个方法不进行给定参数的类型检查,它就是为了避免代码中的强制转换(Casting)。然而这可能会改变(类型检查可以添加)将来的主要版本。
参照Matchers类的javadoc帮助文档中的例子
Parameters:
- keyClazz - map key类型避免类型强制转换(Casting)
- valueClazz - value类型避免类型强制转换(Casting)
Returns:
- 空Map.
anyObject
public static < T> T anyObject()
匹配任何事物, 包含null.
这是: any()和any(java.lang.Class)的别名
参照Matchers类的javadoc帮助文档中的例子
Returns:
- empty null.
anySet**
public static <a href="http://docs.oracle.com/javase/8/docs/api/java/util/Set.html?is-external=true">Set</a> anySet()
任何非空(non-null)Set.
参照Matchers类的javadoc帮助文档中的例子
Returns:
- 空Set.
anySetOf
public static < T> Set < T> anySetOf(Class< T> clazz)
通用友好的别名anySet()。为了保持代码清洁,通过@SuppressWarnings(“unchecked”)来进行替代编译器警告。
任何非空(non-null)Set.
这个方法不进行给定参数的类型检查,它就是为了避免代码中的强制转换(Casting)。然而这可能会改变(类型检查可以添加)将来的主要版本。
参照Matchers类的javadoc帮助文档中的例子
Parameters:
- clazz - 类型属于Set为了避免类型强制转换(Casting)
Returns:
- 空Set.
anyShort
public static short anyShort()
任何short类型或非空(non-null)Short.
参照Matchers类的javadoc帮助文档中的例子
Returns:
anyString
public static String anyString()
任何非空(non-null)String
参照Matchers类的javadoc帮助文档中的例子
Returns:
- 空String (“”).
anyVararg
public static < T> T anyVararg()
任何vararg类型, 即任何参数(arguments)的number和values
例如:
//verification:
mock.foo(1, 2);
mock.foo(1, 2, 3, 4);
verify(mock, times(2)).foo(anyVararg());
//stubbing:
when(mock.foo(anyVararg()).thenReturn(100);
//prints 100
System.out.println(mock.foo(1, 2));
//also prints 100<
System.out.println(mock.foo(1, 2, 3, 4));
参照Matchers类的javadoc帮助文档中的例子
Returns:
- null.
argThat
public static < T> T argThat(ArgumentMatcher < T> matcher)
允许创建自定义的参数匹配模式.这个API在2.0中已经改变,请阅读ArgumentMatcher基础指南。
在实现自定义参数匹配模式前,理解使用的场景和non-trivial参数的可用选项是非常重要的。这种方式下,你可以在给定的情况下选择最好的方法来设计制造高质量的测试(清洁和维护).请阅读ArgumentMatcher文档学习方法和例子。
在极少数情况下,当参数是基本数据类型(primitive)时,你必须使用相关的intThat()、floatThat()等方法。这些方法在进行自动拆箱(auto-unboxing)时可以避免NullPointerException异常。
参照ArgumentMatcher类的javadoc帮助文档中的例子
Parameters:
- matcher - 取决于选择的参数匹配模式(argument matches)
Returns:
- null.
booleanThat
public static boolean booleanThat(ArgumentMatcher < Boolean> matcher)
允许创建自定义的Boolean类型参数匹配模式(Boolean argument matchers).
参照Matchers类的javadoc帮助文档中的例子
Parameters:
-
matcher - 取决于选择的参数匹配模式(argument matches)
Returns: -
false.
byteThat
public static byte byteThat(ArgumentMatcher < Byte> matcher)
允许创建自定义的Byte类型参数匹配模式(Byte argument matchers)
参照Matchers类的javadoc帮助文档中的例子
Parameters:
- matcher - 取决于选择的参数匹配模式(argument matches)
Returns:
charThat
public static char charThat(ArgumentMatcher < Character> matcher)
允许创建自定义的Character类型参数匹配模式(Character argument matchers)
参照Matchers类的javadoc帮助文档中的例子
Parameters:
- matcher - 取决于选择的参数匹配模式(argument matches)
Returns:
contains
public static String contains(String substring)
String参数包含给定的substring字符串.
参照Matchers类的javadoc帮助文档中的例子
Parameters:
- substring - substring字符串.
Returns:
- 空String (“”).
description函数
public static VerificationMod description(String description)
添加验证失败时要输出的文字内容
verify(mock, description("This will print on failure")).someMethod("some arg");
Parameters:
输出的文字内容
Returns:
验证模式
Since:
- 2.0.0
doAnswer函数
public static Stubber doAnswer(Answer answer)
当你想要测试一个无返回值的函数时,可以使用一个含有泛型类Answer参数的doAnswer()函数。为无返回值的函数做测试桩与when(Objecy)
方法不同,因为编译器不喜欢在大括号内使用void函数。
doAnswer(new Answer() {
public Object answer(InvocationOnMock invocation) {
Object[] args = invocation.getArguments();
Mock mock = invocation.getMock();
return null;
}})
.when(mock).someMethod();
参照Mockito类的javadoc帮助文档中的例子
Parameters:
测试函数的应答内容
Returns:
测试方法的测试桩
doCallRealMethod函数
public static Stubber doCallRealMethod()
如果你想调用某一个方法的真实实现请使用doCallRealMethod()
。
像往常一样你需要阅读局部的mock对象警告:面向对象编程通过将复杂的事物分解成单独的、具体的、SRPY对象来减少对复杂事件的处理。
局部模拟是如何符合这种范式的呢。?局部模拟通常情况下是指在对象相同的情况下那些复杂的事物被移动另一个不同的方法中。在大多数情况下,并没有按照你所希望的方式来设计你的应用。
然而,使用局部mock对象也会有个别情况:有些代码你并不能非常容易的改变(3rd接口,临时遗留代码的重构等),但是我对于新的、测试驱动及良好设计的代码不会使用局部mock对象。
同样在javadoc中spy(Object)
关于partial mocks的说明.推荐使用Mockito.spy()
来创建局部mock对象原因是由于你负责构建对象并传值到spy()
中,它只管保证被调用。
Foo mock = mock(Foo.class);
doCallRealMethod().when(mock).someVoidMethod();
// this will call the real implementation of Foo.someVoidMethod()
// 调用Foo.someVoidMethod()的真实现
mock.someVoidMethod();
参照Mockito类的javadoc帮助文档中的例子
Returns:
测试方法的测试桩
Since:
- 1.9.5
doNothing函数
public static Stubber doNothing()
使用doNothing()
函数是为了设置void函数什么也不做。需要注意的是默认情况下返回值为void的函数在mocks中是什么也不做的但是,也会有一些特殊情况。
1.测试桩连续调用一个void函数
doNothing().
doThrow(new RuntimeException())
.when(mock).someVoidMethod();
//does nothing the first time:
//第一次才能都没做
mock.someVoidMethod();
//throws RuntimeException the next time:
//一下次抛出RuntimeException
mock.someVoidMethod();
2.当你监控真实的对象并且你想让void函数什么也不做:
List list = new LinkedList();
List spy = spy(list);
//let's make clear() do nothing
doNothing().when(spy).clear();
spy.add("one");
//clear() does nothing, so the list still contains "one"
spy.clear();
参照Mockito类的javadoc帮助文档中的例子
Returns:
stubber - 测试方法的测试桩
doReturn函数
public static Stubber doReturn(Object toBeReturned)
在某些特殊情况下如果你无法使用when(Object)
可以使用doReturn()
函数
注意:对于测试桩推荐使用when(Object)
函数,因为它是类型安全的并且可读性更强(特别是在测试桩连续调用的情况下)
都有哪些特殊情况下需要使用doReturn()
1.当监控真实的对象并且调用真实的函数带来的影响时:
List list = new LinkedList();
List spy = spy(list);
//Impossible: real method is called so spy.get(0) throws IndexOutOfBoundsException (the list is yet empty)
// 不能完成的:真实方法被调用所以spy.get(0)抛出IndexOutOfBoundsException(list仍是空的)
when(spy.get(0)).thenReturn("foo");
//You have to use doReturn() for stubbing:
//你应用使用doReturn()函数
doReturn("foo").when(spy).get(0);
- 重写一个前exception-stubbing:
when(mock.foo()).thenThrow(new RuntimeException());
//Impossible: the exception-stubbed foo() method is called so RuntimeException is thrown.
// 不能完成的:exception-stubbed foo()被调用抛出RuntimeException异常
when(mock.foo()).thenReturn("bar");
//You have to use doReturn() for stubbing:
//你应用使用doReturn()函数
doReturn("bar").when(mock).foo();
上面的情况展示了Mockito’s的优雅语法。注意这些情况并不常见。监控应该是分散的并且重写exception-stubbing也不常见。更何况对于指出测试桩并复写测试桩是一种潜在的代码嗅觉
参照Mockito类的javadoc帮助文档中的例子
Parameters:
toBeReturned - 当测试桩函数被调用时要被返回的对象
Returns:
stubber - 测试方法的测试桩
doThrow函数
public static Stubber doThrow(Class<? extends Throwable> toBeThrown)
当你想测试void函数中指定类的抛出异常时使用doThrow()
当每一个函数被调用时一个新的异常实例将会被创建
为无返回值的函数做测试桩与when(Objecy)
方法不同,因为编译器不喜欢在大括号内使用void函数。
doThrow(RuntimeException.class).when(mock).someVoidMethod();
Parameters:
测试方法被调用时返回的对象
Returns:
测试方法的测试桩
Since:
- 1.9.0
doThrow函数
public static Stubber doThrow(Throwable toBeThrown)
当测试一个void函数的异常时使用doThrow()
测试void函数需要与使用when(Object)
不同的方式,因为编译器不喜欢大括号内有void函数
Example:
doThrow(RuntimeException.class).when(mock).someVoidMethod();
Parameters:
测试方法被调用时返回的对象
Returns:
测试方法的测试桩
Since:
- 1.9.0
ignoreStubs函数
public static Object[] ignoreStubs(Object... mocks)
忽略对验证函数的测试,当与verifyNoMoreInteractions()
成对出现或是验证inOrder()时是很有用的。避免了在测试时的多余验证,实际上我们对验证测试一点也不感兴趣。
警告,ignoreStubs()
可能会导致verifyNoMoreInteractions(ignoreStubs(...))
的过度使用。考虑到Mockito并不推荐使用verifyNoMoreInteractions()
函数轰炸每一个测试,这其中的原由在文档verifyNoMoreInteractions(Object…)
部分已经说明:换句话说在mocks中所有* stubbed * 的函数都被标记为 * verified * 所以不需要使用这种方式。
该方法改变了input mocks!该方法只是为了方便返回 imput mocks 。
忽略测试也会被忽略掉验证inOrder,包括InOrder.verifyNoMoreInteractions()
,看下面第二个示例:
Example:
//mocking lists for the sake of the example (if you mock List in real you will burn in hell)
List mock1 = mock(List.class), mock2 = mock(List.class);
//stubbing mocks:
when(mock1.get(0)).thenReturn(10);
when(mock2.get(0)).thenReturn(20);
//using mocks by calling stubbed get(0) methods:
// 调用stubbed get(0)使用mocks
System.out.println(mock1.get(0)); //prints 10
System.out.println(mock2.get(0)); //prints 20
//using mocks by calling clear() methods:
// 调用clear()使用mocks
mock1.clear();
mock2.clear();
//verification:
// 验证
verify(mock1).clear();
verify(mock2).clear();
//verifyNoMoreInteractions() fails because get() methods were not accounted for.
// verifyNoMoreInteractions()会失败,因为get()未关联账号
try { verifyNoMoreInteractions(mock1, mock2); } catch (NoInteractionsWanted e);
//However, if we ignore stubbed methods then we can verifyNoMoreInteractions()
//如要我们忽略测试函数我可以这样verifyNoMoreInteractions()
verifyNoMoreInteractions(ignoreStubs(mock1, mock2));
//Remember that ignoreStubs() *changes* the input mocks and returns them for convenience.
忽略测试可以用于verification in order:
List list = mock(List.class);
when(mock.get(0)).thenReturn("foo");
list.add(0);
System.out.println(list.get(0)); //we don't want to verify this
list.clear();
InOrder inOrder = inOrder(ignoreStubs(list));
inOrder.verify(list).add(0);
inOrder.verify(list).clear();
inOrder.verifyNoMoreInteractions();
Parameters:
将被改变的input mocks
Returns:
和传入参数一样的mocks
Since:
1.9.0
inOrder函数
public static InOrder inOrder(Object... mocks)
创建InOrder对象验证 mocks in order
InOrder inOrder = inOrder(firstMock, secondMock);
inOrder.verify(firstMock).add("was called first");
inOrder.verify(secondMock).add("was called second");
验证in order是很灵活的。你可以只验证你感兴趣的,并不需要一个一个验证所有的交互。同样你也可以创建InOrder对象只在相关in-order的验证中进行传值。
InOrder
验证是’greedy’.你很难每一个都注意到。你可以在Mockito wiki pages页搜索’greedy’获取更多信息。
Mockito 1.8.4版本你能以order-sensitive方式调用verifyNoMoreInvocations()
,InOrder.verifyNoMoreInteractions()
参照Mockito类的javadoc帮助文档中的例子
Parameters:
in order中被修改的mocks
Returns:
in order中被用于验证的InOrder对象
mock函数
public static <T> T mock(Class <T> classToMock)
对给定的类或接口创建mock对象
Parameters:
需要mock的类或接口
Returns:
mock对象
mock函数
public static <T> T mock(Class <T> classToMock, Answer defaultAnswer)
根据它对交互的回应指明策略创建mock对象。这个是比较高级特性并且你不需要它写多少测试代码。但是对于legacy系统这是非常有用的。
这个是默认answer,所以当你不想测试函数时你可以使用它。
Foo mock = mock(Foo.class, RETURNS_SMART_NULLS);
Foo mockTwo = mock(Foo.class, new YourOwnAnswer());
参照Mockito类的javadoc帮助文档中的例子
Parameters:
-
需要mock的类或接口
-
未测试函数的默认answer
Returns:
mock对象
mock函数
public static <T> T mock(Class <T> classToMock, MockSettings mockSettings)
没有标准的设置来创建mock对象
配置点的数目对mock的扩大有影响,所以我们在没有越来越多重载Mockito.mock()的情况下需要一种更流利的方式来介绍一种新的配置方式。即MockSettings.
Listener mock = mock(Listener.class, withSettings()
.name("firstListner").defaultBehavior(RETURNS_SMART_NULLS));
);
使用它时需要小心一些并且不要常用。在什么情况下你的测试会不需要标准配置的mocks?在测试代码下太复杂以至于不需要标准配置的mocks?你有没有重构测试代码来让它更加容易测试?
也可以参考withSettings()
参照Mockito类的javadoc帮助文档中的例子
Parameters:
- 需要mock的类或接口
- mock配置
Returns:
- mock对象
mock
@Deprecated
@已过期
public static <T> T mock(Class <T> classToMock, ReturnValues returnValues)
已过期,请使用mock(Foo.class, defaultAnswer)
;
已过期,请使用mock(Foo.class, defaultAnswer)
;
见mock(Class, Answer)
为什么会过期?为了框架更好的一致性与交互性用Answer替换了ReturnValues。Answer接口很早就存在框架中了并且它有和ReturnValues一样的责任。没有必要维护两个一样的接口。
针对它的返回值需要指明策略来创建mock对象。这个是比较高级特性并且你不需要它写多少测试代码。但是对于legacy系统这是非常有用的。
明显地,当你不需要测试方法时可以使用这个返回值。
Foo mock = mock(Foo.class, Mockito.RETURNS_SMART_NULLS);
Foo mockTwo = mock(Foo.class, new YourOwnReturnValues());
参照Mockito类的javadoc帮助文档中的例子
Parameters:
- 需要mock的类或接口
- 未测试方法默认返回值
Returns:
- mock对象
mock
public static <T> T mock(Class <T> classToMock, String name)
指明mock的名字。命名mock在debug的时候非常有用。名字会在所有验证错误中使用。需要注意的是对于使用太多mock或者collaborators的复杂代码命名mock并不能解决问题。如果你使用了太多的mock,为了更加容易测试/调试 你需要对其进行重构而不是对mock命名。
如果你使用了@Mock注解,意味着你的mock已经有名字了!
@Mock使用字段名称作为mock名字Read more
参照Mockito类的javadoc帮助文档中的例子
Parameters:
- 需要mock的类或接口
- mock的名字
Returns:
- mock对象
mockingDetails函数
public static MockingDetails mockingDetails(Object toInspect)
对于Mockito的关联信息返回MockingDetails实例可以用于检查某一特定的对象,无论给定的对象是mock还是监视的都可以被找出来。
在Mockito以后的版本中MockingDetails可能会扩大并且提供其它有用的有关mock的信息。e.g. invocations, stubbing info, etc.
Parameters:
- 要检查的对象。允许为空
Returns:
MockingDetails实例
Since:
- 1.9.5
n-w开头的函数
never()函数
public static VerificationMode never()
相当于times(0)
,可参见 times(int)
验证交互没有发生. 例如:
verify(mock, never()).someMethod();
如果你想验证mock
之间没有交互,可以使用verifyZeroInteractions(Object...)
或者 verifyNoMoreInteractions(Object...)
这两个方法
具体例子可以参考Javadoc
中的Mockito
类
Returns:
- 验证模式
only()函数
public static VerificationMode only()
如果当前mock
的方法只被调用一次,则允许被检验。例如:
verify(mock, only()).someMethod();
//上面这行代码是下面这两行代码的简写
verify(mock).someMethod();
verifyNoMoreInvocations(mock);
可以参考verifyNoMoreInteractions(Object...)
方法
具体例子可以参考Javadoc
中的Mockito
类
Returns:
- verification mode
reset(T… mocks)函数
public static <T> void reset(T... mocks)
聪明的程序员很少会使用这个方法,因为他们知道使用这个方法意味着这个测试写的很low.通常情况下,你不需要重置你的mocks
,你仅仅需要为你的测试方法创建新的mocks
就可以了。
你可以考虑写一些简单的、精悍的、聚焦的测试方法来代替reset()
这个方法。当你在在测试方法的中间部分用到reset()这个方法时,说明你的测试方法太庞大了。
请遵循以下关于测试方法的建议:请保证你的测试方法在一个动作中短小、精悍、聚焦
。在mockito
的邮件列表中有很多关于这方面的主题讨论。
我们添加reset()
方法的唯一原因是使得注入容器的mocks
得以有效的运行,具体可以参看issue 55 here or FAQ here
不要败坏了你在程序员界的名声,测试方法中间的reset()
方法是代码中的害群之马(这意味着你的这个测试方法太多)
List mock = mock(List.class);
when(mock.size()).thenReturn(10);
mock.add(1);
reset(mock);
//此时会清除你之前所有的交互以及测试桩
Type Parameters:
- T - mocks的类型
Parameters:
- 被重置的mocks
spy(Class classToSpy)函数
@Incubating public static <T> T spy(Class<T> classToSpy)
请参考关于类spy
的文档,过渡使用spy
会导致代码变的非常糟糕。
相比与原来的spy
(对象),这种方法可以在类的基础上创建一个spy
,而不是一个对象。有时你可以很方便基于类创建spy而避免提供一个spy
对象的实例。
因为他们不能被实例化,所以这个对于抽象类的监控非常有用。参见mocksettings.useconstructor()
。
例如:
SomeAbstract spy = spy(SomeAbstract.class);
//Robust API, via settings builder:
//稳定的API,充过builder方式来设置
OtherAbstract spy = mock(OtherAbstract.class, withSettings()
.useConstructor().defaultAnswer(CALLS_REAL_METHODS));
//Mocking a non-static inner abstract class:
//模拟一个非静态抽象内部类
InnerAbstract spy = mock(InnerAbstract.class, withSettings()
.useConstructor().outerInstance(outerInstance).defaultAnswer(CALLS_REAL_METHODS));
Type Parameters:
- T - spy的类型
Parameters:
- spy的类
Returns:
- a spy of the provided class
Since:
- 1.10.12
stub(T methodCall)函数
public static <T> DeprecatedOngoingStubbing<T> stub(T methodCall)
对一个方法打桩会返回结果值或者错误异常,例如:
stub(mock.someMethod()).toReturn(10);
//you can use flexible argument matchers, e.g:
//你可以使用灵活的参数匹配,例如:
stub(mock.someMethod(anyString())).toReturn(10);
//setting exception to be thrown:
//设置抛出的异常
stub(mock.someMethod("some arg")).toThrow(new RuntimeException());
//you can stub with different behavior for consecutive method calls.
// 你可以对不同作用的连续回调的方法打测试桩:
//Last stubbing (e.g: toReturn("foo")) determines the behavior for further consecutive calls.
// 最后面的测试桩(例如:返回一个对象:"foo")决定了接下来的回调方法以及它的行为。
stub(mock.someMethod("some arg"))
.toThrow(new RuntimeException())
.toReturn("foo");
一些用户有点混乱、混淆,是因为相比于’stub()','when(Object)'更加被推荐
//Instead of:
//替代为:
stub(mock.count()).toReturn(10);
//你可以这样做:
//You can do:
when(mock.count()).thenReturn(10);
当对一个返回值为空且抛出异常的方法打测试桩:doThrow(Throwable)
测试桩会被重写:例如通常测试桩会设置为常用固定设置,但测试方法可以重写它。切记重写测试桩是一种非常不推荐的写法,因为这样做会导致非常多的测试桩。
一旦这个方法打过桩,无论这个方法被调用多少次,这个方法会一直返回这个测试桩的值。
当你对相同的方法调用相同的参数打测试桩很多次,最后面的测试桩则非常重要
尽管我们可以去验证对测试桩的调用,但通常它都是多余的。比如说你对foo.bar()
打测试桩。如果你比较关心的是当某些情况foo.bar()
中断了(经常在verify()
方法执行之前),此时会返回什么。如果你的代码不关心是get(0)
会返回什么那么它就不应该被添加测试桩。如果你还不确定?看这里
Parameters:
- methodCall - 调用的方法
Returns:
- DeprecatedOngoingStubbing 对象是用来设置测试桩的值或者异常的
stubVoid(T mock)函数
public static <T> VoidMethodStubbable<T> stubVoid(T mock)
已废弃.使用doThrow(Throwable)
方法代替去打空测试桩
//Instead of:
//替代为:
stubVoid(mock).toThrow(e).on().someVoidMethod();
//Please do:
//请这样做:
doThrow(e).when(mock).someVoidMethod();
doThrow()
之所以取代了stubVoid()
方法,是因为它增加了和它的兄弟方法doAnswer()
的可读性以及一致性
stubVoid(mock).toThrow(new RuntimeException()).on().someMethod();
//you can stub with different behavior for consecutive calls.
//你可以对不同作用的连续回调的方法打测试桩:
//Last stubbing (e.g. toReturn()) determines the behavior for further consecutive calls.
//最后面的测试桩(例如:`toReturn()`)决定了接下来的回调方法以及它的行为。
stubVoid(mock)
.toThrow(new RuntimeException())
.toReturn()
.on().someMethod();
具体例子可以参考Javadoc
中的Mockito
类
Parameters:
- mock - to stub
Returns:
- stubbable object that allows stubbing with throwable
timesout(long millis)函数
public static VerificationWithTimeout timeout(long millis)
允许验证时使用timeout
。它会在指定的时间后触发你所期望的动作,而不是立即失败,也许这个对并发条件下的测试非常有用。它和after()
是有所有不同的,因为after()
会等候一个完整的时期,除非最终的测试结果很快就出来了(例如:当never()
失败了), 然而当验证通过时,timeout()
会快速地停止,当你使用times(2)
时会产生不同的行为。例如,当先通过然后失败,在这种情况下,timeout
将会当time(2)
通过时迅速通过,然而after()
将会一直运行直到times(2)
失败,然后它也一同失败。
这个功能看起来应该极少使用,但在多线程的系统的测试中,这是一个很好的方式
目前尚未实现按照顺序去验证
//passes when someMethod() is called within given time span
//当`someMethod()`被以时间段的形式调用时通过
verify(mock, timeout(100)).someMethod();
//above is an alias to:
// 上面的是一个别名
verify(mock, timeout(100).times(1)).someMethod();
//passes as soon as someMethod() has been called 2 times before the given timeout
// 在超时之前,`someMethod()`通过了2次调用
verify(mock, timeout(100).times(2)).someMethod();
//equivalent: this also passes as soon as someMethod() has been called 2 times before the given timeout
//这个和上面的写法是等价的,也是在超时之前,`someMethod()`通过了2次调用
verify(mock, timeout(100).atLeast(2)).someMethod();
//verifies someMethod() within given time span using given verification mode
//在一个超时时间段内,用自定义的验证模式去验证`someMethod()`方法
//useful only if you have your own custom verification modes.
//只有在你有自己定制的验证模式时才有用
verify(mock, new Timeout(100, yourOwnVerificationMode)).someMethod();
具体例子可以参考Javadoc
中的Mockito
类
Parameters:
- millis - - 时间长度(单位:毫秒)
Returns:
- 验证模式
time(int wantedNumberOfInvocations)函数
public static VerificationMode times(int wantedNumberOfInvocations)
允许验证调用方法的精确次数,例如:
verify(mock, times(2)).someMethod("some arg");
//连续调用该方法两次
具体例子可以参考Javadoc
中的Mockito
类
Parameters:
- wantedNumberOfInvocations - 希望调用的次数
Returns:
- 验证模式
validateMockitoUsage()函数
public static void validateMockitoUsage()
首页,无论遇到任何问题,我都鼓励你去阅读the Mockito
问答集:http://groups.google.com/group/mockito,你也可以在mockito
邮件列表提问http://groups.google.com/group/mockito.
validateMockitoUsage()
会明确地检验framework
的状态以用来检查Mockito
是否有效使用。但是,这个功能是可选的,因为**'Mockito`会使这个用法一直有效**,不过有一个问题请继续读下去。
错误示例:
//Oops, thenReturn() part is missing:
//当心,`thenReturn()`部分是缺失的
when(mock.get());
//Oops, verified method call is inside verify() where it should be on the outside:
//当心,下面验证方法的调用在`verify()`里面,其实应该在外面
verify(mock.execute());
//Oops, missing method to verify:
//当心,验证缺失方法
verify(mock);
如果你错误的使用了Mockito
,这样将会抛出异常,这样你就会知道你的测试是否写的正确。你要清楚当你使用这个框架时,Mockito
会接下来的所有时刻开始验证(例如:下一次你验证、打测试桩、调用mock
等)。尽管在下一次测试中可能会抛出异常,但这个异常消息包含了一个完整栈踪迹路径以及这个错误的位置。此时你可以点击并找到这个Mockito
使用错误的地方。
有时,你可能想知道这个框架的使用方法。比如,一个用户想将validateMockitoUsage()
放在它的@after
方法中,为了能快速地知道它使用Mockito
时哪里错了。如果没有这个,它使用这个框架时将不能那么迅速地知道哪里使用错了。另外在@after
中使用validateMockitoUsage()
比较好的一点是jUnit runner
以及Junit rule
中的测试方法在有错误时也会失败,然而普通的next-time
验证可能会在下一次测试方法中才失败。但是尽管Junit
可能对下一次测试报告显示红色,但不要担心,这个异常消息包含了一个完整栈踪迹路径以及这个错误的位置。此时你可以点击并找到这个Mockito
使用错误的地方。
同样在runner中:MockitoJUnitRunner and rule
:MockitoRule
在每次测试方法之后运行validateMockitoUsage()
一定要牢记通常你不需要’validateMockitoUsage()'和框架验证,因为基于next-time
触发的应该已经足够,主要是因为可以点击出错位置查看强大的错误异常消息。但是,如果你已经有足够的测试基础(比如你为所有的测试写有自己的runner
或者基类),我将会推荐你使用validateMockitoUsage()
,因为对@After
添加一个特别的功能时将是零成本。
具体例子可以参考Javadoc
中的Mockito
类
verify(T mock)函数
public static <T> T verify(T mock)
验证发生的某些行为
等同于verify(mock, times(1))
例如:
verify(mock).someMethod("some arg");
上面的写法等同于:
verify(mock, times(1)).someMethod("some arg");
参数比较是通过equals()
方法。可参考ArgumentCaptor
或者ArgumentMatcher
中关于匹配以及断言参数的方法。
尽管我们可以去验证对测试桩的调用,但通常它都是多余的。比如说你对foo.bar()
打测试桩。如果你比较关心的是当某些情况foo.bar()
中断了(经常在verify()
方法执行之前),此时会返回什么。如果你的代码不关心是get(0)
会返回什么那么它就不应该被添加测试桩。如果你还不确定?看这里
具体例子可以参考Javadoc
中的Mockito
类
Parameters:
-
mock - 要被验证的
Returns: -
mock本身
verifyNoMoreInteractions(Object… mocks)函数
public static void verifyNoMoreInteractions(Object... mocks)
检查入参的mocks
是否有任何未经验证的交互,你可以在验证你的mocks
之后使用这个方法,用以确保你的mocks
没有其它地方会被调用.
测试柱的调用也被看成是交互。
**警告:**一些使用者,倾向于经常使用verifyNoMoreInteractions()
方法做大量经典的、期望-运行-验证的模拟,甚至是在所有的测试方法中使用。verifyNoMoreInteractions()
并不被推荐于使用在所有的测试方法中。在交互测试工具中,verifyNoMoreInteractions()
是一个很方便的断言。你只能在当它是明确的、相关的时候使用它。滥用它将导致多余的指定、不可维护的测试。你可以在这里查找更多的文章。
这个方法会在测试方法运行之前检查未经验证的调用,例如:在setUp()
,@Before
方法或者构造函数中。考虑到要写出良好优秀的代码,交互只能够在测试方法中。
示例:
//interactions
//交互
mock.doSomething();
mock.doSomethingUnexpected();
//verification
//验证
verify(mock).doSomething();
//following will fail because 'doSomethingUnexpected()' is unexpected
//因为'doSomethingUnexpected()'是未被期望的,所以下面将会失败
verifyNoMoreInteractions(mock);
具体例子可以参考Javadoc
中的Mockito
类
Parameters:
- mocks - 被验证的
verifyZeroInteractions(Object… mocks)函数
public static void verifyZeroInteractions(Object... mocks)
传进来的mocks之间没有任何交互。
verifyZeroInteractions(mockOne, mockTwo);
这个方法会在测试方法运行之前检查调用,例如:在setUp()
,@Before
方法或者构造函数中。考虑到要写出良好的代码,交互只能够在测试方法中。
你也可以参考never()方法 - 这个方法很明确的表达了当前方法的用途.
具体例子可以参考Javadoc
中的Mockito
类
Parameters:
- mocks - 被验证的
when(T methodCall)函数
public static <T> OngoingStubbing<T> when(T methodCall)
使测试桩方法生效。当你想让这个mock
能调用特定的方法返回特定的值,那么你就可以使用它。
简而言之:当你调用x
方法时会返回y
。
when()
是继承自已经废弃的方法stub(Object)
例如:
when(mock.someMethod()).thenReturn(10);
//you can use flexible argument matchers, e.g:
//你可以使用灵活的参数匹配,例如
when(mock.someMethod(anyString())).thenReturn(10);
//setting exception to be thrown:
//设置抛出的异常
when(mock.someMethod("some arg")).thenThrow(new RuntimeException());
//you can set different behavior for consecutive method calls.
//你可以对不同作用的连续回调的方法打测试桩:
//Last stubbing (e.g: thenReturn("foo")) determines the behavior of further consecutive calls.
//最后面的测试桩(例如:返回一个对象:"foo")决定了接下来的回调方法以及它的行为。
when(mock.someMethod("some arg"))
.thenThrow(new RuntimeException())
.thenReturn("foo");
//Alternative, shorter version for consecutive stubbing:
//可以用以下方式替代,比较小版本的连贯测试桩:
when(mock.someMethod("some arg"))
.thenReturn("one", "two");
//is the same as:
//和下面的方式效果是一样的
when(mock.someMethod("some arg"))
.thenReturn("one")
.thenReturn("two");
//shorter version for consecutive method calls throwing exceptions:
//比较小版本的连贯测试桩并且抛出异常:
when(mock.someMethod("some arg"))
.thenThrow(new RuntimeException(), new NullPointerException();
当你打空方法的测试桩,相关异常可参见:doThrow(Throwable)
,Stubbing
可以被重写:比如:普通的测试桩可以使用固定的设置,但是测试方法能够重写它。切记重写测试桩是一种非常不推荐的写法,因为这样做会导致非常多的测试桩。
一旦这个方法打过桩,无论这个方法被调用多少次,这个方法会一直返回这个测试桩的值。
当你对相同的方法调用相同的参数打测试桩很多次,最后面的测试桩则非常重要
尽管我们可以去验证对测试桩的调用,但通常它都是多余的。比如说你对foo.bar()
打测试桩。如果你比较关心的是当某些情况foo.bar()
中断了(经常在verify()
方法执行之前),此时会返回什么。如果你的代码不关心是get(0)
会返回什么那么它就不应该被添加测试桩。如果你还不确定?看这里
具体例子可以参考Javadoc
中的Mockito
类
Parameters:
- methodCall - 调用的方法
Returns:
- 通常是
OngoingStubbing
对象。不要为被返回的对象创建一个引用。
withSettings()函数
public static MockSettings withSettings()
可以在创建mock
时添加设置。
不要经常去设置它。应该在使用简单的mocks
时写简单的设置。跟着我重复:简单的测试会使整体的代码更简单,更可读、更可维护。如果你不能把测试写的很简单-那么请在测试时重构你的代码。
mock
设置的例子
//Creates mock with different default answer & name
//用不同的默认结果和名字去创建`mock`
Foo mock = mock(Foo.class, withSettings()
.defaultAnswer(RETURNS_SMART_NULLS)
.name("cool mockie"));
//Creates mock with different default answer, descriptive name and extra interfaces
用不同的默认结果和描述的名称以及额外的接口去创建`mock`
Foo mock = mock(Foo.class, withSettings()
.defaultAnswer(RETURNS_SMART_NULLS)
.name("cool mockie")
.extraInterfaces(Bar.class));
有两种原因推荐使用MockSettings
.第一种,有需求要增加另外一种mock
设置,这样用起来更方便。第二种,能够结合不同的moke
设置以减少大量重载moke()
方法。
具体可参考MockSettings
文档来学习mock settins
Returns:
mock settings
默认实例