Mockito如何使用输出参数模拟void方法?

时间:2013-01-29 11:56:56

标签: java mockito

我有一个void方法“functionVoid”,它通知一个参数。

public class MyMotherClass {
 @Inject
 MyClass2 myClass2

 public String motherFunction(){
  ....
  String test = "";
  myClass2.functionVoid(test);

  if (test.equals("")) {
      IllegalArgumentException ile = new IllegalArgumentException(
      "Argument is not valid");
      logger.throwing(ile);
      throw ile;
  }
  ....
 }
}

public class MyClass2 {

public void functionVoid(String output_value)
{ ....
 output_value = "test";

 ....
 }
}

如何在JUnit方法中模拟此方法我的方法“motherFunction”? 在我的例子中,“test”变量仍然是空的。

@RunWith(MockitoJUnitRunner.class)
public class MyMotherClassTest {

 @Mock
 private MyClass2 myClass2 ;

 @InjectMock
 private final MyMotherClass myMotherClass = new MyMotherClass ();

 @Test
 public void test(){

  myMotherClass.motherFunction();     

 }
}

4 个答案:

答案 0 :(得分:17)

如果你想模拟motherFunction的返回结果,那么你不必担心方法的内部实现(最终调用functionVoid)。您需要做的是向 Mockito 提供有关在调用方法motherFunction时该怎么做的说明,这可以通过 来实现带语法的子句;

    when(mockedObject.motherFunction()).thenReturn("Any old string");

如果这忽略了您尝试实现的目标,请查看文档中的how to mock void methods,并确定doAnswer的使用是否适用于此处,例如;

doAnswer(new Answer<Void>() {

  @Override
  public Void answer(InvocationOnMock invocation) throws Throwable {
    String output_value = invocation.getArguments()[0];
    output_value = "Not blank";
    return null;
  }
}).when(myClass2).functionVoid(anyString());

答案 1 :(得分:6)

如果你可以改变functionVoid()来接受一个可变对象作为参数,那么你应该能够达到你想要的效果。

例如,如果您按如下方式更改functionVoid():

public void functionVoid(StringBuilder output_value)
{ ....
 output_value.append("test");

 ....
 }

并在您的motherFunction中调用它,如下所示:

public String motherFunction(){
  ....
  StringBuilder test = new StringBuilder();
  myClass2.functionVoid(test);

  if (test.toString().equals("")) {

现在修改OceanLife上面的答案,您应该可以执行以下操作:

doAnswer(new Answer<Void>() {

  @Override
  public Void answer(InvocationOnMock invocation) throws Throwable {
    StringBuilder output_value = invocation.getArguments()[0];
    output_value.append("Not blank");
    return null;
  }
}).when(myClass2).functionVoid(any(StringBuilder.class)); 

当然,如果你可以改变functionVoid(),你也可以让它返回一个String而不是void。

答案 2 :(得分:4)

  

在我的例子中,“test”变量仍为空。

这不是Mockito问题。

请查看此question,尤其是this answer

它的要点是Java是按值传递的(这在上面的链接中解释得更好)。 Mockito或Java中的任何内容都不能使test var成为空字符串以外的任何内容。它在方法调用之前是一个空字符串,并且在调用之后将是一个空字符串。

您可以在方法中更改对象的状态(例如,将对象添加到方法中的集合中)并在退出方法时查看这些更改,但是您无法更改var在方法中引用的对象并期望这些更改退出方法后“坚持”。然而,字符串实际上是不可变的(没有状态可以改变),所以你甚至不能这样做。

因此,在该方法调用中不能对test进行任何修改。

答案 3 :(得分:0)

如果要检查对象Obj的方法someMethod(String arg),那么:

String argument = "";
Mockito.verify(Obj, Mockito.times(1)).someMethod(argument);
Obj必须是模拟或间谍。 当你想检查是否将正确的参数传递给void方法时,这是有效的。

如果你的方法修改了某种方式,那么你应该使用断言: someMethod(StringWrapper wrapper)改变了字符串。

// given
String argument = "a";
String expected = "a_changed";
String wrapped = new StringWrapper(a);

// when
someMethod(wrapped);

// then
Assert.assertEquals(wrapped.getString(), expected)

我不确定你是否正在寻找这个?