11、为回调做测试桩
when(mock.someMethod(anyString())).thenAnswer(new Answer() {
Object answer(InvocationOnMock invocation) {
Object[] args = invocation.getArguments();
Object mock = invocation.getMock();
return "called with arguments: " + args;
}
});
//调用mock.someMethod("foo")将打印 "called with arguments: foo"
System.out.println(mock.someMethod("foo"));
Answer 是个泛型接口。到调用发生时将执行这个回调,用过 Object[] args = invocation.getArguments();可以拿到调用时传入的参数,通过 Object mock = invocation.getMock();可以拿到mock对象。
有些方法可能接口的参数为一个Listener参数,如果我们使用Answer打桩,我们就可以获取这个Listener,并且在Answer函数中执行对应的回调函数,这对我们了解函数的内部执行过成有很大的帮助。
12、doReturn()|doThrow()| doAnswer()|doNothing()|doCallRealMethod() 系列方法
doThrow(new RuntimeException()).when(mockedList).clear();
//将会 抛出 RuntimeException:
mockedList.clear();
这个实力表示当执行到mockedList.clear()时,将会抛出RuntimeException。其他的doXXX执行与它类似
当需要小面这些功能时这是必须要做的:
(1)、测试void函数
(2)、在受监控的对象上测试函数
(3)、不知一次的测试同一个函数,在测试过程中改变Mock对象的行为
13、监控真实对象
你可以为真实对象创建一个监控(spy)对象,当你使用这个spy对象时,真实的对象也会被调用,除非它的函数被打桩。你应该尽量少的使用spy对象,使用时也需要小心,例如spy对象可以用来处理遗留代码,Spy示例如下:
List list = new LinkedList();
//监控一个真实对象
List spy = spy(list);
//你可以为某些函数打桩
when(spy.size()).thenReturn(100);
//使用这个将调用真实对象的函数
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异常,因为真实对象是空的
when(spy.get(0)).thenReturn("foo");
//你需要使用 doReturn() 来打桩
doReturn("foo").when(spy).get(0);
Mockito并不会为真实的对象代理函数调用,实际上它会复制真实对象,因此,如果你保留了真实对象并且与之交互,不要期望监控对象得到正确的结果。当你在监控对象上调用一个没有stub函数使,并不会调用真实对象的对应函数,你不会在真实对象上看到任何效果。
因此结论就是:当你监控一个真实对象时,你想为这个真实对象的函数做测试桩,那么就是自找麻烦。或者根本不应该验证这些函数。
更多使用请参考:http://mockito.org/