java - 我怎樣才能讓一個方法返回傳遞給它的參數?


考慮一個方法簽名,如:

public String myFunction(String abc);

Mockito可以幫助返回方法收到的相同的字符串嗎?


Answers



您可以在Mockito中創建答案。 我們假設,我們有一個名為Application的接口,其方法為myFunction。

public interface Application {
  public String myFunction(String abc);
}

這是Mockito答案的測試方法:

public void testMyFunction() throws Exception {
  Application mock = mock(Application.class);
  when(mock.myFunction(anyString())).thenAnswer(new Answer<String>() {
    @Override
    public String answer(InvocationOnMock invocation) throws Throwable {
      Object[] args = invocation.getArguments();
      return (String) args[0];
    }
  });

  assertEquals("someString",mock.myFunction("someString"));
  assertEquals("anotherString",mock.myFunction("anotherString"));
}

由於Mockito 1.9.5和Java 8通過使用lambda函數有一個更簡單的方法:

when(myMock.myFunction(anyString())).thenAnswer(i -> i.getArguments()[0]);




如果你有Mockito 1.9.5或更高版本,有一個新的靜態方法可以為你創建Answer對象。 你需要寫一些類似的東西

when(myMock.myFunction(anyString())).then(returnsFirstArg());

或者可選地

doAnswer(returnsFirstArg()).when(myMock).myFunction(anyString());

請注意,在returnsFirstArg()類中, returnsFirstArg()方法是靜態的,這對於Mockito 1.9.5是新的; 所以你需要正確的靜態導入。




使用Java 8,即使使用老版本的Mockito,也可以創建一個單行的答案:

when(myMock.myFunction(anyString()).then(i -> i.getArgumentAt(0, String.class));

當然,這不像使用David Wallace所建議的AdditionalAnswers那樣有用,但是如果你想要“隨時”變換參數,可能會有用。




我有一個非常類似的問題。 我們的目標是嘲笑一個持久化Object的服務,並且可以通過它們的名字來返回它們。 該服務如下所示:

public class RoomService {
    public Room findByName(String roomName) {...}
    public void persist(Room room) {...}
}

服務模擬使用地圖來存儲房間實例。

RoomService roomService = mock(RoomService.class);
final Map<String, Room> roomMap = new HashMap<String, Room>();

// mock for method persist
doAnswer(new Answer<Void>() {
    @Override
    public Void answer(InvocationOnMock invocation) throws Throwable {
        Object[] arguments = invocation.getArguments();
        if (arguments != null && arguments.length > 0 && arguments[0] != null) {
            Room room = (Room) arguments[0];
            roomMap.put(room.getName(), room);
        }
        return null;
    }
}).when(roomService).persist(any(Room.class));

// mock for method findByName
when(roomService.findByName(anyString())).thenAnswer(new Answer<Room>() {
    @Override
    public Room answer(InvocationOnMock invocation) throws Throwable {
        Object[] arguments = invocation.getArguments();
        if (arguments != null && arguments.length > 0 && arguments[0] != null) {
            String key = (String) arguments[0];
            if (roomMap.containsKey(key)) {
                return roomMap.get(key);
            }
        }
        return null;
    }
});

我們現在可以在這個模擬上運行我們的測試。 例如:

String name = "room";
Room room = new Room(name);
roomService.persist(room);
assertThat(roomService.findByName(name), equalTo(room));
assertNull(roomService.findByName("none"));



用Java 8, 史蒂夫的答案可以成為

public void testMyFunction() throws Exception {
    Application mock = mock(Application.class);
    when(mock.myFunction(anyString())).thenAnswer(
    invocation -> {
        Object[] args = invocation.getArguments();
        return args[0];
    });

    assertEquals("someString", mock.myFunction("someString"));
    assertEquals("anotherString", mock.myFunction("anotherString"));
}

編輯:甚至更短:

public void testMyFunction() throws Exception {
    Application mock = mock(Application.class);
    when(mock.myFunction(anyString())).thenAnswer(
        invocation -> invocation.getArgument(0));

    assertEquals("someString", mock.myFunction("someString"));
    assertEquals("anotherString", mock.myFunction("anotherString"));
}



我使用類似的東西(基本上是相同的方法)。 有時,有一個模擬對象返回某些輸入的預定義輸出是有用的。 這是這樣的:

private Hashtable<InputObject,  OutputObject> table = new Hashtable<InputObject, OutputObject>();
table.put(input1, ouput1);
table.put(input2, ouput2);

...

when(mockObject.method(any(InputObject.class))).thenAnswer(
       new Answer<OutputObject>()
       {
           @Override
           public OutputObject answer(final InvocationOnMock invocation) throws Throwable
           {
               InputObject input = (InputObject) invocation.getArguments()[0];
               if (table.containsKey(input))
               {
                   return table.get(input);
               }
               else
               {
                   return null; // alternatively, you could throw an exception
               }
           }
       }
       );



您可能希望將verify()與ArgumentCaptor結合使用以確保在測試中執行,並使用ArgumentCaptor來評估參數:

ArgumentCaptor<String> argument = ArgumentCaptor.forClass(String.class);
verify(mock).myFunction(argument.capture());
assertEquals("the expected value here", argument.getValue());

參數的值顯然可以通過argument.getValue()進一步操作/檢查/不管。