พิจารณาลายเซ็นวิธีการเช่น:
public String myFunction(String abc);
Mockito ช่วยส่งคืนสตริงเดียวกันกับที่ได้รับหรือไม่?
พิจารณาลายเซ็นวิธีการเช่น:
public String myFunction(String abc);
Mockito ช่วยส่งคืนสตริงเดียวกันกับที่ได้รับหรือไม่?
คำตอบ:
คุณสามารถสร้างคำตอบใน Mockito สมมติว่าเรามีอินเทอร์เฟซชื่อแอปพลิเคชันด้วยวิธีการของฟังก์ชัน
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 expression:
when(myMock.myFunction(anyString())).thenAnswer(i -> i.getArguments()[0]);
when(...).then(Return.firstParameter())
when(foo(any()).then(i -> i.getArgumentAt(0, Bar.class))
มันเป็นอาหารมื้อเย็นง่ายที่จะกลับอาร์กิวเมนต์แรกแม้สำหรับเรียนเฉพาะคือ และคุณสามารถใช้วิธีการอ้างอิงและเรียกวิธีการจริงได้เช่นกัน
Iterator<? extends ClassName>
ซึ่งทำให้เกิดปัญหาการโยนทุกชนิดในthenReturn()
คำสั่ง
when(foo(any()).thenAnswer(i -> i.getArguments()[0])
หากคุณมี Mockito 1.9.5 หรือสูงกว่ามีวิธีคงที่ใหม่ที่สามารถสร้างAnswer
วัตถุให้คุณได้ คุณต้องเขียนอะไรซักอย่าง
import static org.mockito.Mockito.when;
import static org.mockito.AdditionalAnswers.returnsFirstArg;
when(myMock.myFunction(anyString())).then(returnsFirstArg());
หรืออีกวิธีหนึ่ง
doAnswer(returnsFirstArg()).when(myMock).myFunction(anyString());
โปรดทราบว่าreturnsFirstArg()
วิธีการนี้เป็นแบบสแตติกในAdditionalAnswers
คลาสซึ่งเป็นสิ่งใหม่สำหรับ Mockito 1.9.5 ดังนั้นคุณจะต้องนำเข้าคงที่ที่เหมาะสม
when(...).then(returnsFirstArg())
ฉันผิดwhen(...).thenReturn(returnsFirstArg())
ที่ให้java.lang.ClassCastException: org.mockito.internal.stubbing.answers.ReturnsArgumentAt cannot be cast to
static org.mockito.AdditionalAnswers.returnsFirstArg
เราจะต้องนำเข้า สิ่งนี้เพื่อใช้ returnFirstArg นอกจากนี้ฉันสามารถทำได้when(myMock.myFunction(any())).then(returnsFirstArg())
ใน Mockito 2.20 *
ด้วย Java 8 เป็นไปได้ที่จะสร้างคำตอบแบบบรรทัดเดียวแม้กับ Mockito เวอร์ชันเก่า:
when(myMock.myFunction(anyString()).then(i -> i.getArgumentAt(0, String.class));
แน่นอนว่ามันไม่ได้มีประโยชน์อย่างที่AdditionalAnswers
เดวิดวอลเลซแนะนำ แต่อาจมีประโยชน์หากคุณต้องการเปลี่ยนการโต้แย้ง "ในทันที"
long
เช่นนี้สิ่งนี้ยังสามารถใช้กับมวยได้Long.class
หรือ
ฉันมีปัญหาที่คล้ายกันมาก เป้าหมายคือการเยาะเย้ยบริการที่ยังคงมีออบเจ็กต์และสามารถคืนพวกเขาด้วยชื่อของพวกเขา บริการมีลักษณะดังนี้:
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 คำตอบของ Steveจะกลายเป็น
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"));
}
นี่เป็นคำถามที่ค่อนข้างเก่า แต่ฉันคิดว่ายังคงเกี่ยวข้อง นอกจากนี้คำตอบที่ได้รับการยอมรับก็ใช้งานได้กับ String เท่านั้น ในขณะเดียวกันก็มี Mockito 2.1 และการนำเข้ามีการเปลี่ยนแปลงดังนั้นฉันต้องการแบ่งปันคำตอบปัจจุบันของฉัน:
import static org.mockito.AdditionalAnswers.returnsFirstArg;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.Mockito.when;
@Mock
private MyClass myClass;
// this will return anything you pass, but it's pretty unrealistic
when(myClass.myFunction(any())).then(returnsFirstArg());
// it is more "life-like" to accept only the right type
when(myClass.myFunction(any(ClassOfArgument.class))).then(returnsFirstArg());
myClass.myFunction จะมีลักษณะดังนี้:
public class MyClass {
public ClassOfArgument myFunction(ClassOfArgument argument){
return argument;
}
}
ฉันใช้สิ่งที่คล้ายกัน (โดยทั่วไปเป็นแนวทางเดียวกัน) บางครั้งมันมีประโยชน์ที่จะให้วัตถุจำลองคืนเอาต์พุตที่กำหนดไว้ล่วงหน้าสำหรับอินพุตบางตัว มันจะเป็นแบบนี้:
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 () สำหรับการจัดการ / ตรวจสอบเพิ่มเติม / อะไรก็ตาม
นี่เก่าไปหน่อย แต่ฉันมาที่นี่เพราะฉันมีปัญหาเดียวกัน ฉันใช้ JUnit แต่คราวนี้ในแอป Kotlin กับ mockk ฉันโพสต์ตัวอย่างที่นี่สำหรับการอ้างอิงและการเปรียบเทียบกับคู่ Java:
@Test
fun demo() {
// mock a sample function
val aMock: (String) -> (String) = mockk()
// make it return the same as the argument on every invocation
every {
aMock.invoke(any())
} answers {
firstArg()
}
// test it
assertEquals("senko", aMock.invoke("senko"))
assertEquals("senko1", aMock.invoke("senko1"))
assertNotEquals("not a senko", aMock.invoke("senko"))
}
คุณสามารถทำได้โดยใช้ArgumentCaptor
ลองนึกภาพคุณมีฟังก์ชั่นถั่วเช่นนั้น
public interface Application {
public String myFunction(String abc);
}
จากนั้นในชั้นทดสอบของคุณ:
//Use ArgumentCaptor to capture the value
ArgumentCaptor<String> param = ArgumentCaptor.forClass(String.class);
when(mock.myFunction(param.capture())).thenAnswer(new Answer<String>() {
@Override
public String answer(InvocationOnMock invocation) throws Throwable {
return param.getValue();//return the captured value.
}
});
หรือถ้าคุณชื่นชอบแลมบ์ดาคุณก็ทำได้:
//Use ArgumentCaptor to capture the value
ArgumentCaptor<String> param = ArgumentCaptor.forClass(String.class);
when(mock.myFunction(param.capture()))
.thenAnswer((invocation) -> param.getValue());
สรุป:ใช้ argumentcaptor เพื่อดักจับพารามิเตอร์ที่ส่งผ่าน ตอบกลับมาคืนค่าที่บันทึกโดยใช้ getValue
This doesn´t work (anymore?).
ฉันได้ทำงานในอินสแตนซ์ของฉันนี้ 2. ขออภัยฉันไม่ชัดเจนในจุดที่คุณพยายามจะทำ คำตอบเฉพาะสำหรับคำถามของ OP